mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
- Fixed behaviour in case the shutdown didn't take place (there were many "next timer event at..." messages in that case). - Reduced the default value for MinEventTimeout to 30 minutes. - Fixed detecting manual start in shutdown feature. - An error message is now displayed in case the Transfer Mode can't be started because the necessary DVB card is currently recording (or there is no DVB card that can access this channel). - Fixed toggling channels with the '0' key in case the "Ok" button has been pressed to display the current/next information. - Pressing the "Power" key now always initiates the shutdown sequence (after user confirmation in case of a recording timer), event if there is currently a menu or a replay session active. Note the additional remarks in INSTALL regarding the values of the two parameters given to the shutdown program in case of a currently recording timer. - Switching through channel groups with the "Left" and "Right" keys now always starts at the group that contains the current channel. - Implemented "Multi Speed Mode" (thanks to Stefan Huelswitt). - Implemented backtracing to hit the right spot after fast forward/rewind (thanks to Stefan Huelswitt). - Implemented replay mode display (thanks to Stefan Huelswitt, with a few rewrites by kls). - Changed the size of all input buffers used to parse config files or receive SVDRP commands to the same value of 10KB. This allows long strings to be used in the 'summary' field of a timer, for instance. - The pipe to the Dolby Digital replay command (option '-a') now closes all unused file descriptors in the child process to avoid crashing when the OSD is used (thanks to Andreas Vitting). - Switched to the driver's new tuning API (VDR now requires a driver version dated 2001-09-14 or higher). - Changed obsolete macro VIDEO_WINDOW_CHROMAKEY to VID_TYPE_CHROMAKEY (thanks to Guido Fiala). - New version of the "Master-Timer" tool (thanks to Matthias Schniedermeyer). - Better error handling when writing configuration files. - Fixed putting the final editing mark into the edited version's marks file. - Fixed manipulating an editing mark at the very end of a recording. - Fixed starting a new replay immediately after stopping a previous one (had caused a mix between live video and replay). - Three new keys ("Volume+", Volume-" and "Mute") to control the DVB card's audio output volume. - New version of the 'epg2timers' tool (thanks to Carsten Koch).
350 lines
13 KiB
C++
350 lines
13 KiB
C++
/*
|
|
* dvbapi.h: Interface to the DVB driver
|
|
*
|
|
* See the main source file 'vdr.c' for copyright information and
|
|
* how to reach the author.
|
|
*
|
|
* $Id: dvbapi.h 1.51 2001/09/16 13:54:23 kls Exp $
|
|
*/
|
|
|
|
#ifndef __DVBAPI_H
|
|
#define __DVBAPI_H
|
|
|
|
#if defined(DEBUG_OSD) || defined(REMOTE_KBD)
|
|
#include <ncurses.h>
|
|
#endif
|
|
#include <stdlib.h> // FIXME: this is apparently necessary for the ost/... header files
|
|
// FIXME: shouldn't every header file include ALL the other header
|
|
// FIXME: files it depends on? The sequence in which header files
|
|
// FIXME: are included here should not matter - and it should NOT
|
|
// FIXME: be necessary to include <stdlib.h> here!
|
|
#include <linux/videodev.h>
|
|
#include <ost/dmx.h>
|
|
#include <ost/sec.h>
|
|
#include <ost/frontend.h>
|
|
#include <ost/video.h>
|
|
#include <ost/audio.h>
|
|
#include <ost/osd.h>
|
|
#include <stdio.h>
|
|
|
|
#include "dvbosd.h"
|
|
#ifdef DVDSUPPORT
|
|
#include "dvd.h"
|
|
#endif //DVDSUPPORT
|
|
#include "eit.h"
|
|
#include "thread.h"
|
|
|
|
// Overlay facilities
|
|
#define MAXCLIPRECTS 100
|
|
typedef struct CRect {
|
|
signed short x, y, width, height;
|
|
};
|
|
|
|
#define FRAMESPERSEC 25
|
|
|
|
// The maximum file size is limited by the range that can be covered
|
|
// with 'int'. 4GB might be possible (if the range is considered
|
|
// 'unsigned'), 2GB should be possible (even if the range is considered
|
|
// 'signed'), so let's use 2000MB for absolute safety (the actual file size
|
|
// may be slightly higher because we stop recording only before the next
|
|
// 'I' frame, to have a complete Group Of Pictures):
|
|
#define MAXVIDEOFILESIZE 2000 // MB
|
|
#define MINVIDEOFILESIZE 100 // MB
|
|
|
|
const char *IndexToHMSF(int Index, bool WithFrame = false);
|
|
// Converts the given index to a string, optionally containing the frame number.
|
|
int HMSFToIndex(const char *HMSF);
|
|
// Converts the given string (format: "hh:mm:ss.ff") to an index.
|
|
|
|
enum eSetChannelResult { scrOk, scrNoTransfer, scrFailed };
|
|
|
|
class cChannel;
|
|
|
|
class cRecordBuffer;
|
|
class cPlayBuffer;
|
|
class cReplayBuffer;
|
|
#ifdef DVDSUPPORT
|
|
class cDVDplayBuffer;
|
|
#endif //DVDSUPPORT
|
|
class cTransferBuffer;
|
|
class cCuttingBuffer;
|
|
|
|
class cVideoCutter {
|
|
private:
|
|
static cCuttingBuffer *cuttingBuffer;
|
|
public:
|
|
static bool Start(const char *FileName);
|
|
static void Stop(void);
|
|
static bool Active(void);
|
|
};
|
|
|
|
class cDvbApi {
|
|
friend class cRecordBuffer;
|
|
friend class cReplayBuffer;
|
|
#ifdef DVDSUPPORT
|
|
friend class cDVDplayBuffer;
|
|
#endif //DVDSUPPORT
|
|
friend class cTransferBuffer;
|
|
private:
|
|
int videoDev;
|
|
int fd_osd, fd_frontend, fd_sec, fd_dvr, fd_audio, fd_video, fd_demuxa1, fd_demuxa2, fd_demuxd1, fd_demuxd2, fd_demuxv, fd_demuxt;
|
|
int vPid, aPid1, aPid2, dPid1, dPid2;
|
|
bool SetPid(int fd, dmxPesType_t PesType, int Pid, dmxOutput_t Output);
|
|
bool SetVpid(int Vpid, dmxOutput_t Output) { return SetPid(fd_demuxv, DMX_PES_VIDEO, Vpid, Output); }
|
|
bool SetApid1(int Apid, dmxOutput_t Output) { return SetPid(fd_demuxa1, DMX_PES_AUDIO, Apid, Output); }
|
|
bool SetApid2(int Apid, dmxOutput_t Output) { return SetPid(fd_demuxa2, DMX_PES_OTHER, Apid, Output); }
|
|
bool SetDpid1(int Dpid, dmxOutput_t Output) { return SetPid(fd_demuxd1, DMX_PES_OTHER, Dpid, Output); }
|
|
bool SetDpid2(int Dpid, dmxOutput_t Output) { return SetPid(fd_demuxd2, DMX_PES_OTHER, Dpid, Output); }
|
|
bool SetTpid(int Tpid, dmxOutput_t Output) { return SetPid(fd_demuxt, DMX_PES_TELETEXT, Tpid, Output); }
|
|
bool SetPids(bool ForRecording);
|
|
cDvbApi(int n);
|
|
public:
|
|
~cDvbApi();
|
|
|
|
#define MAXDVBAPI 4
|
|
static int NumDvbApis;
|
|
private:
|
|
static cDvbApi *dvbApi[MAXDVBAPI];
|
|
static int useDvbApi;
|
|
int cardIndex;
|
|
public:
|
|
static cDvbApi *PrimaryDvbApi;
|
|
static void SetUseDvbApi(int n);
|
|
// Sets the 'useDvbApi' flag of the given DVB device.
|
|
// If this function is not called before Init(), all DVB devices
|
|
// will be used.
|
|
static bool SetPrimaryDvbApi(int n);
|
|
// Sets the primary DVB device to 'n' (which must be in the range
|
|
// 1...NumDvbApis) and returns true if this was possible.
|
|
static cDvbApi *GetDvbApi(int Ca, int Priority);
|
|
// Selects a free DVB device, starting with the highest device number
|
|
// (but avoiding, if possible, the PrimaryDvbApi).
|
|
// If Ca is not 0, the device with the given number will be returned.
|
|
// If all DVB devices are currently recording, the one recording the
|
|
// lowest priority timer (if any) that is lower than the given Priority
|
|
// will be returned.
|
|
// The caller must check whether the returned DVB device is actually
|
|
// recording and stop recording if necessary.
|
|
int CardIndex(void) { return cardIndex; }
|
|
// Returns the card index of this DvbApi (0 ... MAXDVBAPI - 1).
|
|
static bool Probe(const char *FileName);
|
|
// Probes for existing DVB devices.
|
|
static bool Init(void);
|
|
// Initializes the DVB API.
|
|
// Must be called before accessing any DVB functions.
|
|
static void Cleanup(void);
|
|
// Closes down all DVB devices.
|
|
// Must be called at the end of the program.
|
|
|
|
// EIT facilities
|
|
|
|
private:
|
|
cSIProcessor *siProcessor;
|
|
public:
|
|
const cSchedules *Schedules(cThreadLock *ThreadLock) const;
|
|
// Caller must provide a cThreadLock which has to survive the entire
|
|
// time the returned cSchedules is accessed. Once the cSchedules is no
|
|
// longer used, the cThreadLock must be destroyed.
|
|
void SetUseTSTime(bool On) { if (siProcessor) siProcessor->SetUseTSTime(On); }
|
|
|
|
// Image Grab facilities
|
|
|
|
bool GrabImage(const char *FileName, bool Jpeg = true, int Quality = -1, int SizeX = -1, int SizeY = -1);
|
|
|
|
// Overlay facilities
|
|
|
|
private:
|
|
bool ovlStat, ovlGeoSet, ovlFbSet;
|
|
int ovlSizeX, ovlSizeY, ovlPosX, ovlPosY, ovlBpp, ovlPalette, ovlClips, ovlClipCount;
|
|
int ovlFbSizeX, ovlFbSizeY;
|
|
__u16 ovlBrightness, ovlColour, ovlHue, ovlContrast;
|
|
struct video_clip ovlClipRects[MAXCLIPRECTS];
|
|
public:
|
|
bool OvlF(int SizeX, int SizeY, int FbAddr, int Bpp, int Palette);
|
|
bool OvlG(int SizeX, int SizeY, int PosX, int PosY);
|
|
bool OvlC(int ClipCount, CRect *Cr);
|
|
bool OvlP(__u16 Brightness, __u16 Color, __u16 Hue, __u16 Contrast);
|
|
bool OvlO(bool Value);
|
|
|
|
// On Screen Display facilities
|
|
|
|
private:
|
|
enum { charWidth = 12, // average character width
|
|
lineHeight = 27 // smallest text height
|
|
};
|
|
#ifdef DEBUG_OSD
|
|
WINDOW *window;
|
|
enum { MaxColorPairs = 16 };
|
|
int colorPairs[MaxColorPairs];
|
|
void SetColor(eDvbColor colorFg, eDvbColor colorBg = clrBackground);
|
|
#else
|
|
cDvbOsd *osd;
|
|
#endif
|
|
int cols, rows;
|
|
public:
|
|
void Open(int w, int h);
|
|
void Close(void);
|
|
void Clear(void);
|
|
void Fill(int x, int y, int w, int h, eDvbColor color = clrBackground);
|
|
void SetBitmap(int x, int y, const cBitmap &Bitmap);
|
|
void ClrEol(int x, int y, eDvbColor color = clrBackground);
|
|
int CellWidth(void);
|
|
int LineHeight(void);
|
|
int Width(unsigned char c);
|
|
int WidthInCells(const char *s);
|
|
eDvbFont SetFont(eDvbFont Font);
|
|
void Text(int x, int y, const char *s, eDvbColor colorFg = clrWhite, eDvbColor colorBg = clrBackground);
|
|
void Flush(void);
|
|
|
|
// Video format facilities:
|
|
|
|
void SetVideoFormat(videoFormat_t Format);
|
|
|
|
// Channel facilities
|
|
|
|
private:
|
|
int currentChannel;
|
|
public:
|
|
eSetChannelResult SetChannel(int ChannelNumber, int FrequencyMHz, char Polarization, int Diseqc, int Srate, int Vpid, int Apid1, int Apid2, int Dpid1, int Dpid2, int Tpid, int Ca, int Pnr);
|
|
static int CurrentChannel(void) { return PrimaryDvbApi ? PrimaryDvbApi->currentChannel : 0; }
|
|
int Channel(void) { return currentChannel; }
|
|
|
|
// Transfer facilities
|
|
|
|
private:
|
|
cTransferBuffer *transferBuffer;
|
|
cDvbApi *transferringFromDvbApi;
|
|
public:
|
|
bool Transferring(void);
|
|
// Returns true if we are currently transferring video data.
|
|
private:
|
|
cDvbApi *StartTransfer(int TransferToVideoDev);
|
|
// Starts transferring video data from this DVB device to TransferToVideoDev.
|
|
void StopTransfer(void);
|
|
// Stops transferring video data (in case a transfer is currently active).
|
|
|
|
// Record/Replay facilities
|
|
|
|
private:
|
|
cRecordBuffer *recordBuffer;
|
|
cPlayBuffer *replayBuffer;
|
|
int ca;
|
|
int priority;
|
|
int Ca(void) { return ca; }
|
|
// Returns the ca of the current recording session (0..MAXDVBAPI).
|
|
int Priority(void) { return priority; }
|
|
// Returns the priority of the current recording session (0..MAXPRIORITY),
|
|
// or -1 if no recording is currently active.
|
|
int SetModeRecord(void);
|
|
// Initiates recording mode and returns the file handle to read from.
|
|
void SetModeReplay(void);
|
|
void SetModeNormal(bool FromRecording);
|
|
public:
|
|
int SecondsToFrames(int Seconds);
|
|
// Returns the number of frames corresponding to the given number of seconds.
|
|
bool Recording(void);
|
|
// Returns true if we are currently recording.
|
|
bool Replaying(void);
|
|
// Returns true if we are currently replaying.
|
|
bool StartRecord(const char *FileName, int Ca, int Priority);
|
|
// Starts recording the current channel into the given file, with
|
|
// the given ca and priority.
|
|
// In order to be able to record longer movies,
|
|
// a numerical suffix will be appended to the file name. The inital
|
|
// value of that suffix will be larger than any existing file under
|
|
// the given name, thus allowing an interrupted recording to continue
|
|
// gracefully.
|
|
// Returns true if recording was started successfully.
|
|
// If there is already a recording session active, false will be
|
|
// returned.
|
|
void StopRecord(void);
|
|
// Stops the current recording session (if any).
|
|
bool StartReplay(const char *FileName);
|
|
// Starts replaying the given file.
|
|
// If there is already a replay session active, it will be stopped
|
|
// and the new file will be played back.
|
|
#ifdef DVDSUPPORT
|
|
bool StartDVDplay(cDVD *dvd, int TitleID);//XXX dvd parameter necessary???
|
|
// Starts replaying the given TitleID on the DVD.
|
|
#endif //DVDSUPPORT
|
|
void StopReplay(void);
|
|
// Stops the current replay session (if any).
|
|
void Pause(void);
|
|
// Pauses the current replay session, or resumes a paused session.
|
|
void Play(void);
|
|
// Resumes normal replay mode.
|
|
void Forward(void);
|
|
// Runs the current replay session forward at a higher speed.
|
|
void Backward(void);
|
|
// Runs the current replay session backwards at a higher speed.
|
|
void SkipSeconds(int Seconds);
|
|
// Skips the given number of seconds in the current replay session.
|
|
// The sign of 'Seconds' determines the direction in which to skip.
|
|
// Use a very large negative value to go all the way back to the
|
|
// beginning of the recording.
|
|
int SkipFrames(int Frames);
|
|
// Returns the new index into the current replay session after skipping
|
|
// the given number of frames (no actual repositioning is done!).
|
|
// The sign of 'Frames' determines the direction in which to skip.
|
|
bool GetIndex(int &Current, int &Total, bool SnapToIFrame = false);
|
|
// Returns the current and total frame index, optionally snapped to the
|
|
// nearest I-frame.
|
|
bool GetReplayMode(bool &Play, bool &Forward, int &Speed);
|
|
// Returns the current replay mode (if applicable).
|
|
// 'Play' tells whether we are playing or pausing, 'Forward' tells whether
|
|
// we are going forward or backward and 'Speed' is -1 if this is normal
|
|
// play/pause mode, 0 if it is single speed fast/slow forward/back mode
|
|
// and >0 if this is multi speed mode.
|
|
void Goto(int Index, bool Still = false);
|
|
// Positions to the given index and displays that frame as a still picture
|
|
// if Still is true.
|
|
|
|
// Audio track facilities
|
|
|
|
public:
|
|
bool CanToggleAudioTrack(void);
|
|
// Returns true if we are currently replaying and this recording has two
|
|
// audio tracks, or if the current channel has two audio PIDs.
|
|
bool ToggleAudioTrack(void);
|
|
// Toggles the audio track if possible.
|
|
|
|
// Dolby Digital audio facilities
|
|
|
|
private:
|
|
static char *audioCommand;
|
|
public:
|
|
static void SetAudioCommand(const char *Command);
|
|
static const char *AudioCommand(void) { return audioCommand; }
|
|
|
|
// Volume facilities:
|
|
|
|
private:
|
|
bool mute;
|
|
int volume;
|
|
public:
|
|
void ToggleMute(void);
|
|
// Turns the volume off or on.
|
|
void SetVolume(int Volume, bool Absolute = false);
|
|
// Sets the volume to the given value, either absolutely or relative to
|
|
// the current volume.
|
|
};
|
|
|
|
class cEITScanner {
|
|
private:
|
|
enum { ActivityTimeout = 60,
|
|
ScanTimeout = 20
|
|
};
|
|
time_t lastScan, lastActivity;
|
|
int currentChannel, lastChannel;
|
|
int numTransponders, *transponders;
|
|
bool TransponderScanned(cChannel *Channel);
|
|
public:
|
|
cEITScanner(void);
|
|
~cEITScanner();
|
|
bool Active(void) { return currentChannel; }
|
|
void Activity(void);
|
|
void Process(void);
|
|
};
|
|
|
|
#endif //__DVBAPI_H
|