mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
- Removed the '#define FE_CAN_2ND_GEN_MODULATION', since it was wrong and the flag is now in the driver, anyway. - The full-featured DVB cards are now given the TS data directly for replay (thanks to Oliver Endriss for enhancing the av7110 driver to make it replay TS data). The patch from ftp://ftp.cadsoft.de/vdr/Developer/av7110_ts_replay__1.diff implements this change in the driver. The patch av7110_v4ldvb_api5_audiobuf_test_1.diff mentioned in version 1.7.2 is still necessary to avoid audio and video glitches on some channels. - Added a typecast in cUnbufferedFile::Write() to avoid an error message when compiling on 64 bit systems. - Added some missing 'const' statements to cBitmap (thanks to Andreas Regel). - Fixed returning complete PES packets in cTsToPes::GetPes() (thanks to Reinhard Nissl). - Added a missing Detach() in cTransfer::Activate() (thanks to Marco Schlüßler). - Added clearing the TS buffers in cDevice::Detach() (thanks to Marco Schlüßler). - Fixed incrementing the continuity counter in cPatPmtGenerator::GetPmt() (thanks to Johann Friedrichs). - Fixed removing deleted recordings in case there is a problem. Once a recording caused a problem with removing, no others were removed any more and an ongoing recording could fill up the disk and cause other recordings to be deleted automatically (reported by Reinhard Nissl). - Added "DEFINES += -D_FILE_OFFSET_BITS=64 -D_LARGEFILE_SOURCE -D_LARGEFILE64_SOURCE" to Make.config.template (thanks to Johann Friedrichs for pointing this out). Plugin authors should add this line to their Makefile or Make.config if they use file access functions that need special versions for 64 bit offsets. - The new command line option -i can be used to set an "instance id", which will be used to distinguish recordings of the same broadcast made by different instances of VDR (suggested by Frank Schmirler). This replaces the use of the "resume id" that was introduced in version 1.7.3. - Added checking mutexCurrentAudioTrack to cDevice::PlayTs() (thanks to Reinhard Nissl for pointing this out). - Fixed handling the pointer field in cPatPmtParser::ParsePmt() (thanks to Frank Schmirler - sorry I swapped two lines when adopting the original patch). - Checking the remaining packet length after processing the pointer field in cPatPmtParser::ParsePat() and cPatPmtParser::ParsePmt() (suggested by Frank Schmirler). - Checking the pointer field in cPatPmtParser::ParsePmt() only in 'payload start' packets (suggested by Frank Schmirler). - Changed cPatPmtGenerator to make sure the PMT pid doesn't collide with any of the actual pids of the channel. - Fixed cDevice::PlayTsAudio() and made cDevice::PlayTsVideo() return 0 if PlayVideo() didn't play anything. - Added an 'int' typecast to calculations involving FramesPerSecond() to avoid compiler warnings (reported by Winfried Koehler). - Fixed detecting frames for pure audio recordings. - Fixed editing PES recordings. The frame type in the index.vdr file generated for the edited PES recording is set to 1 for I-frames and 2 for all others (P- and B-frames). The exact frame type doesn't matter for VDR, it only needs to know if it's an I-frame or not. - The PAT/PMT is now only processed if its version changes (reported by Reinhard Nissl). - Fixed handling the maximum video file size (reported by Udo Richter). - Improved fast-forward/-rewind for audio recordings. The actual data is now sent to the output device, so that it can be replayed and thus cause the proper delay. For pure audio recordings the audio is no longer muted in fast-forward/-rewind mode, so that some orientation regarding the position within the recording is possible. There may still be some offset in the replay position displayed by the progress indicator when switching from fast-forward/-rewind to play mode, as well as in the current position during normal play mode. This is due to the various buffers between the player and the output device and will be addressed later. Note the new function cDevice::IsPlayingVideo(), which is used to inform the player whether there is video data in the currently replayed stream. If a derived cDevice class reimplements PlayTs() or PlayPes(), it also needs to make sure this new function works as expected.
284 lines
9.7 KiB
C++
284 lines
9.7 KiB
C++
/*
|
|
* recording.h: Recording file handling
|
|
*
|
|
* See the main source file 'vdr.c' for copyright information and
|
|
* how to reach the author.
|
|
*
|
|
* $Id: recording.h 2.4 2009/01/24 15:24:19 kls Exp $
|
|
*/
|
|
|
|
#ifndef __RECORDING_H
|
|
#define __RECORDING_H
|
|
|
|
#include <time.h>
|
|
#include "channels.h"
|
|
#include "config.h"
|
|
#include "epg.h"
|
|
#include "thread.h"
|
|
#include "timers.h"
|
|
#include "tools.h"
|
|
|
|
extern bool VfatFileSystem;
|
|
extern int InstanceId;
|
|
|
|
void RemoveDeletedRecordings(void);
|
|
void AssertFreeDiskSpace(int Priority = 0, bool Force = false);
|
|
///< The special Priority value -1 means that we shall get rid of any
|
|
///< deleted recordings faster than normal (because we're cutting).
|
|
///< If Force is true, the check will be done even if the timeout
|
|
///< hasn't expired yet.
|
|
|
|
class cResumeFile {
|
|
private:
|
|
char *fileName;
|
|
bool isPesRecording;
|
|
public:
|
|
cResumeFile(const char *FileName, bool IsPesRecording);
|
|
~cResumeFile();
|
|
int Read(void);
|
|
bool Save(int Index);
|
|
void Delete(void);
|
|
};
|
|
|
|
class cRecordingInfo {
|
|
friend class cRecording;
|
|
private:
|
|
tChannelID channelID;
|
|
char *channelName;
|
|
const cEvent *event;
|
|
cEvent *ownEvent;
|
|
char *aux;
|
|
double framesPerSecond;
|
|
int priority;
|
|
int lifetime;
|
|
char *fileName;
|
|
cRecordingInfo(const cChannel *Channel = NULL, const cEvent *Event = NULL);
|
|
void SetData(const char *Title, const char *ShortText, const char *Description);
|
|
void SetAux(const char *Aux);
|
|
public:
|
|
cRecordingInfo(const char *FileName);
|
|
~cRecordingInfo();
|
|
tChannelID ChannelID(void) const { return channelID; }
|
|
const char *ChannelName(void) const { return channelName; }
|
|
const char *Title(void) const { return event->Title(); }
|
|
const char *ShortText(void) const { return event->ShortText(); }
|
|
const char *Description(void) const { return event->Description(); }
|
|
const cComponents *Components(void) const { return event->Components(); }
|
|
const char *Aux(void) const { return aux; }
|
|
double FramesPerSecond(void) const { return framesPerSecond; }
|
|
void SetFramesPerSecond(double FramesPerSecond);
|
|
bool Read(FILE *f);
|
|
bool Write(FILE *f, const char *Prefix = "") const;
|
|
bool Read(void);
|
|
bool Write(void) const;
|
|
};
|
|
|
|
class cRecording : public cListObject {
|
|
friend class cRecordings;
|
|
private:
|
|
mutable int resume;
|
|
mutable char *titleBuffer;
|
|
mutable char *sortBuffer;
|
|
mutable char *fileName;
|
|
mutable char *name;
|
|
mutable int fileSizeMB;
|
|
int channel;
|
|
int instanceId;
|
|
bool isPesRecording;
|
|
double framesPerSecond;
|
|
cRecordingInfo *info;
|
|
cRecording(const cRecording&); // can't copy cRecording
|
|
cRecording &operator=(const cRecording &); // can't assign cRecording
|
|
static char *StripEpisodeName(char *s);
|
|
char *SortName(void) const;
|
|
int GetResume(void) const;
|
|
public:
|
|
time_t start;
|
|
int priority;
|
|
int lifetime;
|
|
time_t deleted;
|
|
cRecording(cTimer *Timer, const cEvent *Event);
|
|
cRecording(const char *FileName);
|
|
virtual ~cRecording();
|
|
virtual int Compare(const cListObject &ListObject) const;
|
|
const char *Name(void) const { return name; }
|
|
const char *FileName(void) const;
|
|
const char *Title(char Delimiter = ' ', bool NewIndicator = false, int Level = -1) const;
|
|
const cRecordingInfo *Info(void) const { return info; }
|
|
const char *PrefixFileName(char Prefix);
|
|
int HierarchyLevels(void) const;
|
|
void ResetResume(void) const;
|
|
double FramesPerSecond(void) { return framesPerSecond; }
|
|
bool IsNew(void) const { return GetResume() <= 0; }
|
|
bool IsEdited(void) const;
|
|
bool IsPesRecording(void) const { return isPesRecording; }
|
|
bool WriteInfo(void);
|
|
bool Delete(void);
|
|
// Changes the file name so that it will no longer be visible in the "Recordings" menu
|
|
// Returns false in case of error
|
|
bool Remove(void);
|
|
// Actually removes the file from the disk
|
|
// Returns false in case of error
|
|
bool Undelete(void);
|
|
// Changes the file name so that it will be visible in the "Recordings" menu again and
|
|
// not processed by cRemoveDeletedRecordingsThread.
|
|
// Returns false in case of error
|
|
};
|
|
|
|
class cRecordings : public cList<cRecording>, public cThread {
|
|
private:
|
|
static char *updateFileName;
|
|
bool deleted;
|
|
time_t lastUpdate;
|
|
int state;
|
|
const char *UpdateFileName(void);
|
|
void Refresh(bool Foreground = false);
|
|
void ScanVideoDir(const char *DirName, bool Foreground = false, int LinkLevel = 0);
|
|
protected:
|
|
void Action(void);
|
|
public:
|
|
cRecordings(bool Deleted = false);
|
|
virtual ~cRecordings();
|
|
bool Load(void) { return Update(true); }
|
|
///< Loads the current list of recordings and returns true if there
|
|
///< is anything in it (for compatibility with older plugins - use
|
|
///< Update(true) instead).
|
|
bool Update(bool Wait = false);
|
|
///< Triggers an update of the list of recordings, which will run
|
|
///< as a separate thread if Wait is false. If Wait is true, the
|
|
///< function returns only after the update has completed.
|
|
///< Returns true if Wait is true and there is anyting in the list
|
|
///< of recordings, false otherwise.
|
|
void TouchUpdate(void);
|
|
///< Touches the '.update' file in the video directory, so that other
|
|
///< instances of VDR that access the same video directory can be triggered
|
|
///< to update their recordings list.
|
|
bool NeedsUpdate(void);
|
|
void ChangeState(void) { state++; }
|
|
bool StateChanged(int &State);
|
|
void ResetResume(const char *ResumeFileName = NULL);
|
|
cRecording *GetByName(const char *FileName);
|
|
void AddByName(const char *FileName, bool TriggerUpdate = true);
|
|
void DelByName(const char *FileName);
|
|
int TotalFileSizeMB(void); ///< Only for deleted recordings!
|
|
};
|
|
|
|
extern cRecordings Recordings;
|
|
extern cRecordings DeletedRecordings;
|
|
|
|
#define DEFAULTFRAMESPERSECOND 25.0
|
|
|
|
class cMark : public cListObject {
|
|
private:
|
|
double framesPerSecond;
|
|
public:
|
|
int position;
|
|
char *comment;
|
|
cMark(int Position = 0, const char *Comment = NULL, double FramesPerSecond = DEFAULTFRAMESPERSECOND);
|
|
virtual ~cMark();
|
|
cString ToText(void);
|
|
bool Parse(const char *s);
|
|
bool Save(FILE *f);
|
|
};
|
|
|
|
class cMarks : public cConfig<cMark> {
|
|
private:
|
|
double framesPerSecond;
|
|
public:
|
|
bool Load(const char *RecordingFileName, double FramesPerSecond = DEFAULTFRAMESPERSECOND, bool IsPesRecording = false);
|
|
void Sort(void);
|
|
cMark *Add(int Position);
|
|
cMark *Get(int Position);
|
|
cMark *GetPrev(int Position);
|
|
cMark *GetNext(int Position);
|
|
};
|
|
|
|
#define RUC_BEFORERECORDING "before"
|
|
#define RUC_AFTERRECORDING "after"
|
|
#define RUC_EDITEDRECORDING "edited"
|
|
|
|
class cRecordingUserCommand {
|
|
private:
|
|
static const char *command;
|
|
public:
|
|
static void SetCommand(const char *Command) { command = Command; }
|
|
static void InvokeCommand(const char *State, const char *RecordingFileName);
|
|
};
|
|
|
|
// The maximum size of a single frame (up to HDTV 1920x1080):
|
|
#define MAXFRAMESIZE KILOBYTE(512)
|
|
|
|
// The maximum file size is limited by the range that can be covered
|
|
// with a 40 bit 'unsigned int', which is 1TB. The actual maximum value
|
|
// used is 6MB below the theoretical maximum, to have some safety (the
|
|
// actual file size may be slightly higher because we stop recording only
|
|
// before the next independent frame, to have a complete Group Of Pictures):
|
|
#define MAXVIDEOFILESIZETS 1048570 // MB
|
|
#define MAXVIDEOFILESIZEPES 2000 // MB
|
|
#define MINVIDEOFILESIZE 100 // MB
|
|
#define MAXVIDEOFILESIZEDEFAULT MAXVIDEOFILESIZEPES
|
|
|
|
struct tIndexTs;
|
|
|
|
class cIndexFile {
|
|
private:
|
|
int f;
|
|
char *fileName;
|
|
int size, last;
|
|
tIndexTs *index;
|
|
bool isPesRecording;
|
|
cResumeFile resumeFile;
|
|
cMutex mutex;
|
|
void ConvertFromPes(tIndexTs *IndexTs, int Count);
|
|
void ConvertToPes(tIndexTs *IndexTs, int Count);
|
|
bool CatchUp(int Index = -1);
|
|
public:
|
|
cIndexFile(const char *FileName, bool Record, bool IsPesRecording = false);
|
|
~cIndexFile();
|
|
bool Ok(void) { return index != NULL; }
|
|
bool Write(bool Independent, uint16_t FileNumber, off_t FileOffset);
|
|
bool Get(int Index, uint16_t *FileNumber, off_t *FileOffset, bool *Independent = NULL, int *Length = NULL);
|
|
int GetNextIFrame(int Index, bool Forward, uint16_t *FileNumber = NULL, off_t *FileOffset = NULL, int *Length = NULL, bool StayOffEnd = false);
|
|
int Get(uint16_t FileNumber, off_t FileOffset);
|
|
int Last(void) { CatchUp(); return last; }
|
|
int GetResume(void) { return resumeFile.Read(); }
|
|
bool StoreResume(int Index) { return resumeFile.Save(Index); }
|
|
bool IsStillRecording(void);
|
|
};
|
|
|
|
class cFileName {
|
|
private:
|
|
cUnbufferedFile *file;
|
|
int fileNumber;
|
|
char *fileName, *pFileNumber;
|
|
bool record;
|
|
bool blocking;
|
|
bool isPesRecording;
|
|
public:
|
|
cFileName(const char *FileName, bool Record, bool Blocking = false, bool IsPesRecording = false);
|
|
~cFileName();
|
|
const char *Name(void) { return fileName; }
|
|
int Number(void) { return fileNumber; }
|
|
cUnbufferedFile *Open(void);
|
|
void Close(void);
|
|
cUnbufferedFile *SetOffset(int Number, off_t Offset = 0);
|
|
cUnbufferedFile *NextFile(void);
|
|
};
|
|
|
|
cString IndexToHMSF(int Index, bool WithFrame = false, double FramesPerSecond = DEFAULTFRAMESPERSECOND);
|
|
// Converts the given index to a string, optionally containing the frame number.
|
|
int HMSFToIndex(const char *HMSF, double FramesPerSecond = DEFAULTFRAMESPERSECOND);
|
|
// Converts the given string (format: "hh:mm:ss.ff") to an index.
|
|
int SecondsToFrames(int Seconds, double FramesPerSecond = DEFAULTFRAMESPERSECOND);
|
|
// Returns the number of frames corresponding to the given number of seconds.
|
|
|
|
int ReadFrame(cUnbufferedFile *f, uchar *b, int Length, int Max);
|
|
|
|
char *ExchangeChars(char *s, bool ToFileSystem);
|
|
// Exchanges the characters in the given string to or from a file system
|
|
// specific representation (depending on ToFileSystem). The given string will
|
|
// be modified and may be reallocated if more space is needed. The return
|
|
// value points to the resulting string, which may be different from s.
|
|
|
|
#endif //__RECORDING_H
|