mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
- In order to avoid problems on NPTL systems, VDR now checks for the presence of NPTL at program start, and if it is, exits and tells the user to do 'export LD_ASSUME_KERNEL=2.4.1' before starting VDR. - Revisited the "Fixed missing audio after replaying a DVD" change because it introduced a sound disturbance when switching between channels on the same transponder (thanks to Marco Schlüßler). - In order to avoid problems on UTF-8 systems, VDR now checks for the presence of UTF-8 at program start, and if it is, exits and tells the user to turn off UTF-8 before starting VDR (thanks to Ludwig Nussel for pointing out a problem with systems that are set to use UTF-8). There are also problems in case the video partition is mounted with "iocharset=utf8" (thanks to Jörg Knitter for reporting this one). Please also read the "IMPORTANT NOTES" section in the INSTALL file! - Some changes to the SPU decoder interface (thanks to Sven Goethel). - Some improvements in cOsd creation (thanks to some suggestions by Jouni Karvo). - Fixed calculating the OSD width and height (thanks to Olaf Henkel for reporting a problem with long event texts in the "Classic VDR" skin). - Fixed switching channels while an encrypted channel is being recorded, because the channel was switched if the new channel was on the same transponder and was a radio channel or an unencrypted channel (thanks to Martin Dauskardt for reporting this one). - No longer using the external 'find' command to scan the video directory for recordings (based on a suggestion by Mirko Dölle). - The list of recordings is now kept statically in memory to avoid long delays when opening the "Recordings" menu. As a side effect, external modifications to the video directory are no longer immediately reflected in the "Recordings" menu. If a plugin manipulates the video directory in any way, it can call the function Recordings.TriggerUpdate() to trigger an update of the list of recordings. If some external tool manipulates the video directory, it can touch the file '.update' in the video directory to trigger an update of the list of recordings. - Fixed a memory leak in theme description handling (thanks to Sascha Volkenandt). - Added cDevice::Flush() to make sure that all data in the video card's buffers has been processed (thanks to Reinhard Nissl). Currently this is not yet actually implemented for FF DVB cards. - Fixed handling the color button texts in cMenuEditStrItem (thanks to Maynard Cedric for reporting this one). - Fixed the description of cRingBufferLinear (thanks to Ludwig Nussel for pointing out this one). - Fixed cRingBufferLinear::Get() in case the buffer wraps around (thanks to Ludwig Nussel for reporting this one).
121 lines
3.6 KiB
C++
121 lines
3.6 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 1.15 2004/06/19 10:32:15 kls Exp $
|
|
*/
|
|
|
|
#ifndef __RINGBUFFER_H
|
|
#define __RINGBUFFER_H
|
|
|
|
#include "thread.h"
|
|
#include "tools.h"
|
|
|
|
class cRingBuffer {
|
|
private:
|
|
cMutex mutex;
|
|
cCondVar readyForPut, readyForGet;
|
|
cMutex putMutex, getMutex;
|
|
int putTimeout;
|
|
int getTimeout;
|
|
int size;
|
|
time_t lastOverflowReport;
|
|
int overflowCount;
|
|
int overflowBytes;
|
|
protected:
|
|
int maxFill;//XXX
|
|
int lastPercent;
|
|
bool statistics;//XXX
|
|
void WaitForPut(void);
|
|
void WaitForGet(void);
|
|
void EnablePut(void);
|
|
void EnableGet(void);
|
|
virtual void Clear(void) = 0;
|
|
virtual int Available(void) = 0;
|
|
int Free(void) { return size - Available() - 1; }
|
|
void Lock(void) { mutex.Lock(); }
|
|
void Unlock(void) { mutex.Unlock(); }
|
|
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 {
|
|
private:
|
|
int margin, head, tail;
|
|
int lastGet;
|
|
uchar *buffer;
|
|
pthread_t getThreadTid;
|
|
public:
|
|
cRingBufferLinear(int Size, int Margin = 0, bool Statistics = false);
|
|
///< 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.
|
|
virtual ~cRingBufferLinear();
|
|
virtual int Available(void);
|
|
virtual void Clear(void);
|
|
///< Immediately clears the ring buffer.
|
|
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 retrieved 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;
|
|
public:
|
|
cFrame(const uchar *Data, int Count, eFrameType = ftUnknown, int Index = -1);
|
|
///< 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; }
|
|
};
|
|
|
|
class cRingBufferFrame : public cRingBuffer {
|
|
private:
|
|
cFrame *head;
|
|
int currentFill;
|
|
void Delete(cFrame *Frame);
|
|
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
|