mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
- Completed the Finnish OSD texts (thanks to Rolf Ahrenberg). - Fixed some descriptor handling in 'libsi' (thanks to Stéphane Esté-Gracias). - Fixed handling the current menu item (thanks to Marc Hoppe). - Fixed assigning events to timers (they no longer get "stuck"). - Added log entries whenever the running status of an event changes (currently only logging the first 30 channels). - Fixed handling timers in VPS margin if the EPG scan is turned on (the EPG scan switched the device away from the channel, so it wouldn't see the change of the running status). - Fixed handling "itemized" texts in EPG data (thanks to Stéphane Esté-Gracias for pointing out this problem, and Marcel Wiesweg for improving 'libsi'). - Fixed handling VPS times at year boundaries. - Avoiding too many consecutive "ring buffer overflow" messages (which only slowed down performance even more). - Taking the Sid into account when detecting version changes in processing the PMT (thanks to Stéphane Esté-Gracias for pointing out this problem). - Completed the Russian OSD texts (thanks to Vyacheslav Dikonov). - Any newline characters in the 'description' of EPG events are now preserved to allow texts to be displayed the way the tv stations have formatted them. This was also necessary to better display itemized texts. - Fixed detecting the running status in case an empty EPG event is broadcast (thanks to Michael Pennewiß for pointing this out). - Improved performance when paging through very long menu lists. - Removed cSchedule::GetEventNumber() and cSchedule::NumEvents(). There is now cSchedule::Events() that returns the list of events directly. - Avoiding occasional bad responsiveness to user interaction caused by assigning events to timers. - Now explicitly turning on the LNB power at startup, because newer drivers don't do this any more (thanks to Oliver Endriss for pointing this out).
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.14 2004/03/07 13:40:45 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 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
|