mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
VDR developer version 2.1.7 is now available at ftp://ftp.tvdr.de/vdr/Developer/vdr-2.1.7.tar.bz2 A 'diff' against the previous version is available at ftp://ftp.tvdr.de/vdr/Developer/vdr-2.1.6-2.1.7.diff MD5 checksums: 1c954bad31ce74cd1cbd7987e62d2a98 vdr-2.1.7.tar.bz2 88a90327a75833b7723942d3bd25f954 vdr-2.1.6-2.1.7.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: - No longer logging an error message in DirSizeMB() if the given directory doesn't exist. This avoids lots of log entries in case several VDRs use the same video directory and one of them has already physically removed a recording directory, while the others still have it in their list of deleted recordings. - Updated the Italian OSD texts (thanks to Diego Pierotto). - A cCamSlot that has WantsTsData set to true in its constructor now also gets the CAT and EMM PIDs data. - Fixed a possible division by zero in frame rate detection. - VDR now reads command line options from *.conf files in /etc/vdr/conf.d (thanks to Lars Hanisch). See vdr.1 and vdr.5 for details. - Fixed a possible crash in the LCARS skin (thanks to Thomas Reufer). - Updated the dvbhddevice plugin source. - Fixed a bug in the Makefile when installing plugins with LCLBLD=1 (thanks to Stefan Huelswitt). - The pid of the PMT in which the CA descriptors of a given channel are broadcast is now stored together with the CA descriptors and can be retrieved by calling GetPmtPid() (this information is only required to receive encrypted channels with the OctopusNet receiver via the 'satip' plugin). - Channels that are not listed in the SDT are now only marked as OBSOLETE if "Setup/DVB/Update channels" is set to a value other than "no" or "PIDs only". - Fixed multiple OBSOLETE marks in channels that are not listed in the SDT in case "Setup/Miscellaneous/Show channel names with source" is set to "yes". - The new function cOsd::DrawScaledBitmap() is now used for drawing subtitles. This function can be reimplemented by high level OSDs which may be able to do the scaling in hardware or otherwise more efficiently (thanks to Thomas Reufer). - Fixed detaching receivers from devices in case a CAM needs to receive the TS (reported by Dietmar Spingler). - Fixed resetting the receiver for EMM pids for CAMs that need to receive the TS (reported by Dietmar Spingler). - Fixed (well, actually worked around) a problem with subtitles not being displayed because the broadcaster doesn't set the data's version numbers as required by the DVB standard (thanks to Rolf Ahrenberg). - Fixed support for systemd (thanks to Christopher Reimer). - Added a missing backslash to the help text of the SVDRP command MOVR (thanks to Lars Hanisch). - Added subsystem id support for DVB devices connected via USB (thanks to Jose Alberto Reguero). - Added the functions IndexOf(), InsertUnique(), AppendUnique() and RemoveElement() to the cVector class (thanks to Stefan Schallenberg). - Fixed a possible out-of-bounds access in cVector::Remove(). - Added functions to set and retrieve the priority of a cReceiver (suggested by Frank Schmirler). - Added the new parameters "Setup/Miscellaneous/Volume steps" and ".../Volume linearize" (thanks to Claus Muus). See the MANUAL for details. - Fixed jumping to an absolute position via the Red key in case replay was paused (reported by Dieter Ferdinand). - Changed the German weekday names from "MonDieMitDonFreSamSon" to "Mo.Di.Mi.Do.Fr.Sa.So." (thanks to Stefan Blochberger). - Now handling CAT sections that consist of more than one TS packet. - Added handling for DTS audio tracks to cPatPmtParser::ParsePmt() (thanks to Thomas Reufer). - Added support for PGS subtitles (thanks to Thomas Reufer). - Use of the function cOsd::GetBitmap() outside of derived classes is now deprecated, and it may be made 'protected' in a future version, since it doesn't work with TrueColor OSDs. Plugin authors may want to modify their code so that it works without this function. - Modified the descriptions of several threads, so that the important information (like device or frontend numbers) is within the first 15 characters of the string, because only these are displayed in thread listings. Plugin authors may want to do the same. - Added the channel name to log messages that reference a channel (suggested by Dietmar Spingler). - Modified the CAM API so that it is possible to implement CAMs that can be freely assigned to any devices (thanks to Jasmin Jessich). - Plugins can now implement the function SetMenuSortMode() in their skin objects derived from cSkinDisplayMenu, to get informed about the currently used sort mode, if applicable (suggested by Martin Schirrmacher). - Added cOsdProvider::OsdSizeChanged(), which plugins that implement an output device can call to signal a change in the OSD that requires a redraw of the currently displayed object (thanks to Thomas Reufer). - Added a comment to cRecorder::Activate() about the need to call Detach() in the destructor (suggested by Eike Sauer). - Now returning from removing deleted recordings after at most 10 seconds, or if the user presses a remote control key, to keep the system from getting unresponsive when removing a huge number of files (reported by Dieter Ferdinand). - Fixed generating the index file of an existing recording in case at the of a TS file there is less data in the buffer than needed by the frame detector. In such a case it was possible that frames were missed, and there was most likely a distortion when replaying that part of a recording. This is mostly a problem for recordings that consist of more than one *.ts file. Single file recordings could only lose some frames at their very end, which probably doesn't matter. At any rate, if you have generated an index file with VDR version 2.0.6, 2.1.5 or 2.1.6, you may want to do so again with this version to make sure the index is OK. - Added the new command line option --updindex, which can be used to update an incomplete index of a recording (based on a patch from Helmut Auer).
312 lines
14 KiB
C++
312 lines
14 KiB
C++
/*
|
|
* ci.h: Common Interface
|
|
*
|
|
* See the main source file 'vdr.c' for copyright information and
|
|
* how to reach the author.
|
|
*
|
|
* $Id: ci.h 3.9 2015/01/15 09:18:09 kls Exp $
|
|
*/
|
|
|
|
#ifndef __CI_H
|
|
#define __CI_H
|
|
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
#include "channels.h"
|
|
#include "thread.h"
|
|
#include "tools.h"
|
|
|
|
#define MAX_CAM_SLOTS_PER_ADAPTER 8 // maximum possible value is 255
|
|
#define MAX_CONNECTIONS_PER_CAM_SLOT 8 // maximum possible value is 254
|
|
#define CAM_READ_TIMEOUT 50 // ms
|
|
|
|
class cCiMMI;
|
|
|
|
class cCiMenu {
|
|
friend class cCamSlot;
|
|
friend class cCiMMI;
|
|
private:
|
|
enum { MAX_CIMENU_ENTRIES = 64 }; ///< XXX is there a specified maximum?
|
|
cCiMMI *mmi;
|
|
cMutex *mutex;
|
|
bool selectable;
|
|
char *titleText;
|
|
char *subTitleText;
|
|
char *bottomText;
|
|
char *entries[MAX_CIMENU_ENTRIES];
|
|
int numEntries;
|
|
bool AddEntry(char *s);
|
|
cCiMenu(cCiMMI *MMI, bool Selectable);
|
|
public:
|
|
~cCiMenu();
|
|
const char *TitleText(void) { return titleText; }
|
|
const char *SubTitleText(void) { return subTitleText; }
|
|
const char *BottomText(void) { return bottomText; }
|
|
const char *Entry(int n) { return n < numEntries ? entries[n] : NULL; }
|
|
int NumEntries(void) { return numEntries; }
|
|
bool Selectable(void) { return selectable; }
|
|
void Select(int Index);
|
|
void Cancel(void);
|
|
void Abort(void);
|
|
bool HasUpdate(void);
|
|
};
|
|
|
|
class cCiEnquiry {
|
|
friend class cCamSlot;
|
|
friend class cCiMMI;
|
|
private:
|
|
cCiMMI *mmi;
|
|
cMutex *mutex;
|
|
char *text;
|
|
bool blind;
|
|
int expectedLength;
|
|
cCiEnquiry(cCiMMI *MMI);
|
|
public:
|
|
~cCiEnquiry();
|
|
const char *Text(void) { return text; }
|
|
bool Blind(void) { return blind; }
|
|
int ExpectedLength(void) { return expectedLength; }
|
|
void Reply(const char *s);
|
|
void Cancel(void);
|
|
void Abort(void);
|
|
};
|
|
|
|
class cDevice;
|
|
class cCamSlot;
|
|
|
|
enum eModuleStatus { msNone, msReset, msPresent, msReady };
|
|
|
|
class cCiAdapter : public cThread {
|
|
friend class cCamSlot;
|
|
private:
|
|
cCamSlot *camSlots[MAX_CAM_SLOTS_PER_ADAPTER];
|
|
void AddCamSlot(cCamSlot *CamSlot);
|
|
///< Adds the given CamSlot to this CI adapter.
|
|
protected:
|
|
cCamSlot *ItCamSlot(int &Iter);
|
|
///< Iterates over all added CAM slots of this adapter. Iter has to be
|
|
///< initialized to 0 and is required to store the iteration state.
|
|
///< Returns NULL if no further CAM slot is found.
|
|
virtual void Action(void);
|
|
///< Handles the attached CAM slots in a separate thread.
|
|
///< The derived class must call the Start() function to
|
|
///< actually start CAM handling.
|
|
virtual int Read(uint8_t *Buffer, int MaxLength) { return 0; }
|
|
///< Reads one chunk of data into the given Buffer, up to MaxLength bytes.
|
|
///< If no data is available immediately, wait for up to CAM_READ_TIMEOUT.
|
|
///< Returns the number of bytes read (in case of an error it will also
|
|
///< return 0).
|
|
virtual void Write(const uint8_t *Buffer, int Length) {}
|
|
///< Writes Length bytes of the given Buffer.
|
|
virtual bool Reset(int Slot) { return false; }
|
|
///< Resets the CAM in the given Slot.
|
|
///< Returns true if the operation was successful.
|
|
virtual eModuleStatus ModuleStatus(int Slot) { return msNone; }
|
|
///< Returns the status of the CAM in the given Slot.
|
|
virtual bool Assign(cDevice *Device, bool Query = false) { return false; }
|
|
///< Assigns this adapter to the given Device, if this is possible.
|
|
///< If Query is 'true', the adapter only checks whether it can be
|
|
///< assigned to the Device, but doesn't actually assign itself to it.
|
|
///< Returns true if the adapter can be assigned to the Device.
|
|
///< If Device is NULL, the adapter will be unassigned from any
|
|
///< device it was previously assigned to. The value of Query
|
|
///< is ignored in that case, and this function always returns
|
|
///< 'true'.
|
|
public:
|
|
cCiAdapter(void);
|
|
virtual ~cCiAdapter();
|
|
///< The derived class must call Cancel(3) in its destructor.
|
|
};
|
|
|
|
class cTPDU;
|
|
class cCiTransportConnection;
|
|
class cCiSession;
|
|
class cCiCaProgramData;
|
|
class cCaPidReceiver;
|
|
|
|
class cCamSlot : public cListObject {
|
|
friend class cCiAdapter;
|
|
friend class cCiTransportConnection;
|
|
private:
|
|
cMutex mutex;
|
|
cCondVar processed;
|
|
cCiAdapter *ciAdapter;
|
|
cDevice *assignedDevice;
|
|
cCaPidReceiver *caPidReceiver;
|
|
int slotIndex;
|
|
int slotNumber;
|
|
cCiTransportConnection *tc[MAX_CONNECTIONS_PER_CAM_SLOT + 1]; // connection numbering starts with 1
|
|
eModuleStatus lastModuleStatus;
|
|
time_t resetTime;
|
|
cTimeMs moduleCheckTimer;
|
|
bool resendPmt;
|
|
int source;
|
|
int transponder;
|
|
cList<cCiCaProgramData> caProgramList;
|
|
const int *GetCaSystemIds(void);
|
|
void SendCaPmt(uint8_t CmdId);
|
|
void NewConnection(void);
|
|
void DeleteAllConnections(void);
|
|
void Process(cTPDU *TPDU = NULL);
|
|
void Write(cTPDU *TPDU);
|
|
cCiSession *GetSessionByResourceId(uint32_t ResourceId);
|
|
public:
|
|
cCamSlot(cCiAdapter *CiAdapter, bool WantsTsData = false);
|
|
///< Creates a new CAM slot for the given CiAdapter.
|
|
///< The CiAdapter will take care of deleting the CAM slot,
|
|
///< so the caller must not delete it!
|
|
///< If WantsTsData is true, the device this CAM slot is assigned to will
|
|
///< call the Decrypt() function of this CAM slot, presenting it the complete
|
|
///< TS data stream of the encrypted programme, including the CA pids.
|
|
virtual ~cCamSlot();
|
|
bool Assign(cDevice *Device, bool Query = false);
|
|
///< Assigns this CAM slot to the given Device, if this is possible.
|
|
///< If Query is 'true', the CI adapter of this slot only checks whether
|
|
///< it can be assigned to the Device, but doesn't actually assign itself to it.
|
|
///< Returns true if this slot can be assigned to the Device.
|
|
///< If Device is NULL, the slot will be unassigned from any
|
|
///< device it was previously assigned to. The value of Query
|
|
///< is ignored in that case, and this function always returns
|
|
///< 'true'.
|
|
cDevice *Device(void) { return assignedDevice; }
|
|
///< Returns the device this CAM slot is currently assigned to.
|
|
bool WantsTsData(void) const { return caPidReceiver != NULL; }
|
|
///< Returns true if this CAM slot wants to receive the TS data through
|
|
///< its Decrypt() function.
|
|
int SlotIndex(void) { return slotIndex; }
|
|
///< Returns the index of this CAM slot within its CI adapter.
|
|
///< The first slot has an index of 0.
|
|
int SlotNumber(void) { return slotNumber; }
|
|
///< Returns the number of this CAM slot within the whole system.
|
|
///< The first slot has the number 1.
|
|
virtual bool Reset(void);
|
|
///< Resets the CAM in this slot.
|
|
///< Returns true if the operation was successful.
|
|
virtual eModuleStatus ModuleStatus(void);
|
|
///< Returns the status of the CAM in this slot.
|
|
virtual const char *GetCamName(void);
|
|
///< Returns the name of the CAM in this slot, or NULL if there is
|
|
///< no ready CAM in this slot.
|
|
virtual bool Ready(void);
|
|
///< Returns 'true' if the CAM in this slot is ready to decrypt.
|
|
virtual bool HasMMI(void);
|
|
///< Returns 'true' if the CAM in this slot has an active MMI.
|
|
virtual bool HasUserIO(void);
|
|
///< Returns true if there is a pending user interaction, which shall
|
|
///< be retrieved via GetMenu() or GetEnquiry().
|
|
virtual bool EnterMenu(void);
|
|
///< Requests the CAM in this slot to start its menu.
|
|
virtual cCiMenu *GetMenu(void);
|
|
///< Gets a pending menu, or NULL if there is no menu.
|
|
virtual cCiEnquiry *GetEnquiry(void);
|
|
///< Gets a pending enquiry, or NULL if there is no enquiry.
|
|
int Priority(void);
|
|
///< Returns the priority if the device this slot is currently assigned
|
|
///< to, or IDLEPRIORITY if it is not assigned to any device.
|
|
virtual bool ProvidesCa(const int *CaSystemIds);
|
|
///< Returns true if the CAM in this slot provides one of the given
|
|
///< CaSystemIds. This doesn't necessarily mean that it will be
|
|
///< possible to actually decrypt such a programme, since CAMs
|
|
///< usually advertise several CA system ids, while the actual
|
|
///< decryption is controlled by the smart card inserted into
|
|
///< the CAM.
|
|
virtual void AddPid(int ProgramNumber, int Pid, int StreamType);
|
|
///< Adds the given PID information to the list of PIDs. A later call
|
|
///< to SetPid() will (de)activate one of these entries.
|
|
virtual void SetPid(int Pid, bool Active);
|
|
///< Sets the given Pid (which has previously been added through a
|
|
///< call to AddPid()) to Active. A later call to StartDecrypting() will
|
|
///< send the full list of currently active CA_PMT entries to the CAM.
|
|
virtual void AddChannel(const cChannel *Channel);
|
|
///< Adds all PIDs if the given Channel to the current list of PIDs.
|
|
///< If the source or transponder of the channel are different than
|
|
///< what was given in a previous call to AddChannel(), any previously
|
|
///< added PIDs will be cleared.
|
|
virtual bool CanDecrypt(const cChannel *Channel);
|
|
///< Returns true if there is a CAM in this slot that is able to decrypt
|
|
///< the given Channel (or at least claims to be able to do so).
|
|
///< Since the QUERY/REPLY mechanism for CAMs is pretty unreliable (some
|
|
///< CAMs don't reply to queries at all), we always return true if the
|
|
///< CAM is currently not decrypting anything. If there is already a
|
|
///< channel being decrypted, a call to CanDecrypt() checks whether the
|
|
///< CAM can also decrypt the given channel. Only CAMs that have replied
|
|
///< to the initial QUERY will perform this check at all. CAMs that never
|
|
///< replied to the initial QUERY are assumed not to be able to handle
|
|
///< more than one channel at a time.
|
|
virtual void StartDecrypting(void);
|
|
///< Triggers sending all currently active CA_PMT entries to the CAM,
|
|
///< so that it will start decrypting.
|
|
virtual void StopDecrypting(void);
|
|
///< Clears the list of CA_PMT entries and tells the CAM to stop decrypting.
|
|
virtual bool IsDecrypting(void);
|
|
///< Returns true if the CAM in this slot is currently used for decrypting.
|
|
virtual uchar *Decrypt(uchar *Data, int &Count);
|
|
///< If this is a CAM slot that can be freely assigned to any device,
|
|
///< but will not be directly inserted into the full TS data stream
|
|
///< in hardware, it can implement this function to be given access
|
|
///< to the data in the device's TS buffer. Data points to a buffer
|
|
///< of Count bytes of TS data. The first byte in Data is guaranteed
|
|
///< to be a TS_SYNC_BYTE.
|
|
///< There are three possible ways a CAM can handle decryption:
|
|
///< 1. If the full TS data is physically routed through the CAM in hardware,
|
|
///< there is no need to reimplement this function.
|
|
///< The default implementation simply sets Count to TS_SIZE and returns Data.
|
|
///< 2. If the CAM works directly on Data and decrypts the TS "in place" it
|
|
///< shall decrypt at least the very first TS packet in Data, set Count to
|
|
///< TS_SIZE and return Data. It may decrypt as many TS packets in Data as it
|
|
///< wants, but it must decrypt at least the very first TS packet (if at all
|
|
///< possible - if, for whatever reasons, it can't decrypt the very first
|
|
///< packet, it must return it regardless). Only this very first TS packet will
|
|
///< be further processed after the call to this function. The next call will
|
|
///< be done with Data pointing to the TS packet immediately following the
|
|
///< previous one.
|
|
///< 3. If the CAM needs to copy the data into a buffer of its own, and/or send
|
|
///< the data to some file handle for processing and later retrieval, it shall
|
|
///< set Count to the number of bytes it has read from Data and return a pointer
|
|
///< to the next available decrypted TS packet (which will *not* be in the
|
|
///< memory area pointed to by Data, but rather in some buffer that is under
|
|
///< the CAM's control). If no decrypted TS packet is currently available, NULL
|
|
///< shall be returned. If no data from Data can currently be processed, Count
|
|
///< shall be set to 0 and the same Data pointer will be offered in the next
|
|
///< call to Decrypt().
|
|
///< A derived class that implements this function will also need
|
|
///< to set the WantsTsData parameter in the call to the base class
|
|
///< constructor to true in order to receive the TS data.
|
|
};
|
|
|
|
class cCamSlots : public cList<cCamSlot> {
|
|
public:
|
|
bool WaitForAllCamSlotsReady(int Timeout = 0);
|
|
///< Waits until all CAM slots have become ready, or the given Timeout
|
|
///< (seconds) has expired. While waiting, the Ready() function of each
|
|
///< CAM slot is called in turn, until they all return true.
|
|
///< Returns true if all CAM slots have become ready within the given
|
|
///< timeout.
|
|
};
|
|
|
|
extern cCamSlots CamSlots;
|
|
|
|
class cChannelCamRelation;
|
|
|
|
class cChannelCamRelations : public cList<cChannelCamRelation> {
|
|
private:
|
|
cMutex mutex;
|
|
cChannelCamRelation *GetEntry(tChannelID ChannelID);
|
|
cChannelCamRelation *AddEntry(tChannelID ChannelID);
|
|
time_t lastCleanup;
|
|
void Cleanup(void);
|
|
public:
|
|
cChannelCamRelations(void);
|
|
void Reset(int CamSlotNumber);
|
|
bool CamChecked(tChannelID ChannelID, int CamSlotNumber);
|
|
bool CamDecrypt(tChannelID ChannelID, int CamSlotNumber);
|
|
void SetChecked(tChannelID ChannelID, int CamSlotNumber);
|
|
void SetDecrypt(tChannelID ChannelID, int CamSlotNumber);
|
|
void ClrChecked(tChannelID ChannelID, int CamSlotNumber);
|
|
void ClrDecrypt(tChannelID ChannelID, int CamSlotNumber);
|
|
};
|
|
|
|
extern cChannelCamRelations ChannelCamRelations;
|
|
|
|
#endif //__CI_H
|