mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
- Fixed a hangup when replaying a TS recording with subtitles activated (reported by Timo Helkio). - Fixed handling the 'new' indicator in the recordings menu for TS recordings (thanks to Derek Kelly). - Added cap_sys_nice to the capabilities that are not dropped (thanks to Rolf Ahrenberg). - Updated the Italian OSD texts (thanks to Diego Pierotto). - Added cRecordingInfo::GetEvent() (thanks to Marcel Unbehaun). - Improved synchronizing the progress display, trick modes and subtitle display to the actual audio/video. This now works independent of any buffer sizes the output device might use. + The cBackTrace class has been replaced with cPtsIndex, which keeps track of the PTS timestamps of recently played frames. + cDevice::GetSTC() is now required to deliver the STC even in trick modes. It is sufficient if it returns the PTS of the most recently presented audio/video frame. + The full-featured DVB cards need an improved firmware in order to return proper STC values in trick modes (thanks to Oliver Endriss for enhancing the av7110 firmware). - Adapted cFrameDetector::Analyze() to HD NTSC broadcasts that split frames over several payload units (thanks to Derek Kelly for reporting this and helping in testing). - Modified cFrameDetector::Analyze() to make it process whole frames at once, so that file I/O overhead is minimized during recording (reported by Günter Niedermeier). - Added command line help for the '-i' option. - Fixed cDvbPlayer::NextFile() to handle files larger than 2GB (thanks to Jose Alberto Reguero). - Improved replay at the begin and end of a recording. The very first and very last frame is now sent to the output device repeatedly until GetSTC() reports that it has been played. cDvbPlayer::Action() no longer calls DeviceFlush() (thanks to Reinhard Nissl for making sure vdr-xine no longer needs this). - Added missing '[]' to the delete operator in cMenuEditStrItem::~cMenuEditStrItem(). - Added missing virtual destructor to cPalette. - Now freeing configDirectory before setting it to a new value in cPlugin::SetConfigDirectory(). - Fixed a crash when jumping to an editing mark in an audio recording. - Fixed the 'VideoOnly' condition in the PlayPes() and PlayTs() calls in cDvbPlayer::Action() (thanks to Reinhard Nissl). - cDevice::PlayTs() now plays as many TS packets as possible in one call. - Making sure any floating point numbers written use a decimal point (thanks to Oliver Endriss for pointing out a problem with the F record in the info file of a recording). - Fixed detecting the frame rate for radio recordings. - Added missing AUDIO_PAUSE/AUDIO_CONTINUE calls to cDvbDevice (thanks to Oliver Endriss). - No longer writing the video type into channels.conf if VPID is 0 (thanks to Oliver Endriss for reporting this). - Improved efficiency of cEIT::cEIT() (thanks to Tobias Bratfisch).
492 lines
12 KiB
C
492 lines
12 KiB
C
/*
|
|
* plugin.c: The VDR plugin interface
|
|
*
|
|
* See the main source file 'vdr.c' for copyright information and
|
|
* how to reach the author.
|
|
*
|
|
* $Id: plugin.c 2.1 2009/04/05 10:16:48 kls Exp $
|
|
*/
|
|
|
|
#include "plugin.h"
|
|
#include <ctype.h>
|
|
#include <dirent.h>
|
|
#include <dlfcn.h>
|
|
#include <stdlib.h>
|
|
#include <time.h>
|
|
#include "config.h"
|
|
#include "interface.h"
|
|
#include "thread.h"
|
|
|
|
#define LIBVDR_PREFIX "libvdr-"
|
|
#define SO_INDICATOR ".so."
|
|
|
|
#define MAXPLUGINARGS 1024
|
|
#define HOUSEKEEPINGDELTA 10 // seconds
|
|
|
|
// --- cPlugin ---------------------------------------------------------------
|
|
|
|
char *cPlugin::configDirectory = NULL;
|
|
|
|
cPlugin::cPlugin(void)
|
|
{
|
|
name = NULL;
|
|
started = false;
|
|
}
|
|
|
|
cPlugin::~cPlugin()
|
|
{
|
|
}
|
|
|
|
void cPlugin::SetName(const char *s)
|
|
{
|
|
name = s;
|
|
I18nRegister(name);
|
|
}
|
|
|
|
const char *cPlugin::CommandLineHelp(void)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
bool cPlugin::ProcessArgs(int argc, char *argv[])
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool cPlugin::Initialize(void)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool cPlugin::Start(void)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
void cPlugin::Stop(void)
|
|
{
|
|
}
|
|
|
|
void cPlugin::Housekeeping(void)
|
|
{
|
|
}
|
|
|
|
void cPlugin::MainThreadHook(void)
|
|
{
|
|
}
|
|
|
|
cString cPlugin::Active(void)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
time_t cPlugin::WakeupTime(void)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
const char *cPlugin::MainMenuEntry(void)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
cOsdObject *cPlugin::MainMenuAction(void)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
cMenuSetupPage *cPlugin::SetupMenu(void)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
bool cPlugin::SetupParse(const char *Name, const char *Value)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
void cPlugin::SetupStore(const char *Name, const char *Value)
|
|
{
|
|
Setup.Store(Name, Value, this->Name());
|
|
}
|
|
|
|
void cPlugin::SetupStore(const char *Name, int Value)
|
|
{
|
|
Setup.Store(Name, Value, this->Name());
|
|
}
|
|
|
|
bool cPlugin::Service(const char *Id, void *Data)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
const char **cPlugin::SVDRPHelpPages(void)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
cString cPlugin::SVDRPCommand(const char *Command, const char *Option, int &ReplyCode)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
void cPlugin::RegisterI18n(const void *)
|
|
{
|
|
dsyslog("plugin '%s' called obsolete function RegisterI18n()", Name());
|
|
}
|
|
|
|
void cPlugin::SetConfigDirectory(const char *Dir)
|
|
{
|
|
free(configDirectory);
|
|
configDirectory = strdup(Dir);
|
|
}
|
|
|
|
const char *cPlugin::ConfigDirectory(const char *PluginName)
|
|
{
|
|
static cString buffer;
|
|
if (!cThread::IsMainThread())
|
|
esyslog("ERROR: plugin '%s' called cPlugin::ConfigDirectory(), which is not thread safe!", PluginName ? PluginName : "<no name given>");
|
|
buffer = cString::sprintf("%s/plugins%s%s", configDirectory, PluginName ? "/" : "", PluginName ? PluginName : "");
|
|
return MakeDirs(buffer, true) ? *buffer : NULL;
|
|
}
|
|
|
|
// --- cDll ------------------------------------------------------------------
|
|
|
|
cDll::cDll(const char *FileName, const char *Args)
|
|
{
|
|
fileName = strdup(FileName);
|
|
args = Args ? strdup(Args) : NULL;
|
|
handle = NULL;
|
|
plugin = NULL;
|
|
}
|
|
|
|
cDll::~cDll()
|
|
{
|
|
delete plugin;
|
|
if (handle)
|
|
dlclose(handle);
|
|
free(args);
|
|
free(fileName);
|
|
}
|
|
|
|
static char *SkipQuote(char *s)
|
|
{
|
|
char c = *s;
|
|
strcpy(s, s + 1);
|
|
while (*s && *s != c) {
|
|
if (*s == '\\')
|
|
strcpy(s, s + 1);
|
|
if (*s)
|
|
s++;
|
|
}
|
|
if (*s) {
|
|
strcpy(s, s + 1);
|
|
return s;
|
|
}
|
|
esyslog("ERROR: missing closing %c", c);
|
|
fprintf(stderr, "vdr: missing closing %c\n", c);
|
|
return NULL;
|
|
}
|
|
|
|
bool cDll::Load(bool Log)
|
|
{
|
|
if (Log)
|
|
isyslog("loading plugin: %s", fileName);
|
|
if (handle) {
|
|
esyslog("attempt to load plugin '%s' twice!", fileName);
|
|
return false;
|
|
}
|
|
handle = dlopen(fileName, RTLD_NOW);
|
|
const char *error = dlerror();
|
|
if (!error) {
|
|
void *(*creator)(void);
|
|
creator = (void *(*)(void))dlsym(handle, "VDRPluginCreator");
|
|
if (!(error = dlerror()))
|
|
plugin = (cPlugin *)creator();
|
|
}
|
|
if (!error) {
|
|
if (plugin && args) {
|
|
int argc = 0;
|
|
char *argv[MAXPLUGINARGS];
|
|
char *p = skipspace(stripspace(args));
|
|
char *q = NULL;
|
|
bool done = false;
|
|
while (!done) {
|
|
if (!q)
|
|
q = p;
|
|
switch (*p) {
|
|
case '\\': strcpy(p, p + 1);
|
|
if (*p)
|
|
p++;
|
|
else {
|
|
esyslog("ERROR: missing character after \\");
|
|
fprintf(stderr, "vdr: missing character after \\\n");
|
|
return false;
|
|
}
|
|
break;
|
|
case '"':
|
|
case '\'': if ((p = SkipQuote(p)) == NULL)
|
|
return false;
|
|
break;
|
|
default: if (!*p || isspace(*p)) {
|
|
done = !*p;
|
|
*p = 0;
|
|
if (q) {
|
|
if (argc < MAXPLUGINARGS - 1)
|
|
argv[argc++] = q;
|
|
else {
|
|
esyslog("ERROR: plugin argument list too long");
|
|
fprintf(stderr, "vdr: plugin argument list too long\n");
|
|
return false;
|
|
}
|
|
q = NULL;
|
|
}
|
|
}
|
|
if (!done)
|
|
p = *p ? p + 1 : skipspace(p + 1);
|
|
}
|
|
}
|
|
argv[argc] = NULL;
|
|
if (argc)
|
|
plugin->SetName(argv[0]);
|
|
optind = 0; // to reset the getopt() data
|
|
return !Log || !argc || plugin->ProcessArgs(argc, argv);
|
|
}
|
|
}
|
|
else {
|
|
esyslog("ERROR: %s", error);
|
|
fprintf(stderr, "vdr: %s\n", error);
|
|
}
|
|
return !error && plugin;
|
|
}
|
|
|
|
// --- cPluginManager --------------------------------------------------------
|
|
|
|
cPluginManager *cPluginManager::pluginManager = NULL;
|
|
|
|
cPluginManager::cPluginManager(const char *Directory)
|
|
{
|
|
directory = NULL;
|
|
lastHousekeeping = time(NULL);
|
|
nextHousekeeping = -1;
|
|
if (pluginManager) {
|
|
fprintf(stderr, "vdr: attempt to create more than one plugin manager - exiting!\n");
|
|
exit(2);
|
|
}
|
|
SetDirectory(Directory);
|
|
pluginManager = this;
|
|
}
|
|
|
|
cPluginManager::~cPluginManager()
|
|
{
|
|
Shutdown();
|
|
free(directory);
|
|
if (pluginManager == this)
|
|
pluginManager = NULL;
|
|
}
|
|
|
|
void cPluginManager::SetDirectory(const char *Directory)
|
|
{
|
|
free(directory);
|
|
directory = Directory ? strdup(Directory) : NULL;
|
|
}
|
|
|
|
void cPluginManager::AddPlugin(const char *Args)
|
|
{
|
|
if (strcmp(Args, "*") == 0) {
|
|
cReadDir d(directory);
|
|
struct dirent *e;
|
|
while ((e = d.Next()) != NULL) {
|
|
if (strstr(e->d_name, LIBVDR_PREFIX) == e->d_name) {
|
|
char *p = strstr(e->d_name, SO_INDICATOR);
|
|
if (p) {
|
|
*p = 0;
|
|
p += strlen(SO_INDICATOR);
|
|
if (strcmp(p, APIVERSION) == 0) {
|
|
char *name = e->d_name + strlen(LIBVDR_PREFIX);
|
|
if (strcmp(name, "*") != 0) { // let's not get into a loop!
|
|
AddPlugin(e->d_name + strlen(LIBVDR_PREFIX));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
char *s = strdup(skipspace(Args));
|
|
char *p = strchr(s, ' ');
|
|
if (p)
|
|
*p = 0;
|
|
dlls.Add(new cDll(cString::sprintf("%s/%s%s%s%s", directory, LIBVDR_PREFIX, s, SO_INDICATOR, APIVERSION), Args));
|
|
free(s);
|
|
}
|
|
|
|
bool cPluginManager::LoadPlugins(bool Log)
|
|
{
|
|
for (cDll *dll = dlls.First(); dll; dll = dlls.Next(dll)) {
|
|
if (!dll->Load(Log))
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool cPluginManager::InitializePlugins(void)
|
|
{
|
|
for (cDll *dll = dlls.First(); dll; dll = dlls.Next(dll)) {
|
|
cPlugin *p = dll->Plugin();
|
|
if (p) {
|
|
isyslog("initializing plugin: %s (%s): %s", p->Name(), p->Version(), p->Description());
|
|
if (!p->Initialize())
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool cPluginManager::StartPlugins(void)
|
|
{
|
|
for (cDll *dll = dlls.First(); dll; dll = dlls.Next(dll)) {
|
|
cPlugin *p = dll->Plugin();
|
|
if (p) {
|
|
isyslog("starting plugin: %s", p->Name());
|
|
if (!p->Start())
|
|
return false;
|
|
p->started = true;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void cPluginManager::Housekeeping(void)
|
|
{
|
|
if (time(NULL) - lastHousekeeping > HOUSEKEEPINGDELTA) {
|
|
if (++nextHousekeeping >= dlls.Count())
|
|
nextHousekeeping = 0;
|
|
cDll *dll = dlls.Get(nextHousekeeping);
|
|
if (dll) {
|
|
cPlugin *p = dll->Plugin();
|
|
if (p) {
|
|
p->Housekeeping();
|
|
}
|
|
}
|
|
lastHousekeeping = time(NULL);
|
|
}
|
|
}
|
|
|
|
void cPluginManager::MainThreadHook(void)
|
|
{
|
|
for (cDll *dll = pluginManager->dlls.First(); dll; dll = pluginManager->dlls.Next(dll)) {
|
|
cPlugin *p = dll->Plugin();
|
|
if (p)
|
|
p->MainThreadHook();
|
|
}
|
|
}
|
|
|
|
bool cPluginManager::Active(const char *Prompt)
|
|
{
|
|
if (pluginManager) {
|
|
for (cDll *dll = pluginManager->dlls.First(); dll; dll = pluginManager->dlls.Next(dll)) {
|
|
cPlugin *p = dll->Plugin();
|
|
if (p) {
|
|
cString s = p->Active();
|
|
if (!isempty(*s)) {
|
|
if (!Prompt || !Interface->Confirm(cString::sprintf("%s - %s", *s, Prompt)))
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
cPlugin *cPluginManager::GetNextWakeupPlugin(void)
|
|
{
|
|
cPlugin *NextPlugin = NULL;
|
|
if (pluginManager) {
|
|
time_t Now = time(NULL);
|
|
time_t Next = 0;
|
|
for (cDll *dll = pluginManager->dlls.First(); dll; dll = pluginManager->dlls.Next(dll)) {
|
|
cPlugin *p = dll->Plugin();
|
|
if (p) {
|
|
time_t t = p->WakeupTime();
|
|
if (t > Now && (!Next || t < Next)) {
|
|
Next = t;
|
|
NextPlugin = p;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return NextPlugin;
|
|
}
|
|
|
|
bool cPluginManager::HasPlugins(void)
|
|
{
|
|
return pluginManager && pluginManager->dlls.Count();
|
|
}
|
|
|
|
cPlugin *cPluginManager::GetPlugin(int Index)
|
|
{
|
|
cDll *dll = pluginManager ? pluginManager->dlls.Get(Index) : NULL;
|
|
return dll ? dll->Plugin() : NULL;
|
|
}
|
|
|
|
cPlugin *cPluginManager::GetPlugin(const char *Name)
|
|
{
|
|
if (pluginManager && Name) {
|
|
for (cDll *dll = pluginManager->dlls.First(); dll; dll = pluginManager->dlls.Next(dll)) {
|
|
cPlugin *p = dll->Plugin();
|
|
if (p && strcmp(p->Name(), Name) == 0)
|
|
return p;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
cPlugin *cPluginManager::CallFirstService(const char *Id, void *Data)
|
|
{
|
|
if (pluginManager) {
|
|
for (cDll *dll = pluginManager->dlls.First(); dll; dll = pluginManager->dlls.Next(dll)) {
|
|
cPlugin *p = dll->Plugin();
|
|
if (p && p->Service(Id, Data))
|
|
return p;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
bool cPluginManager::CallAllServices(const char *Id, void *Data)
|
|
{
|
|
bool found=false;
|
|
if (pluginManager) {
|
|
for (cDll *dll = pluginManager->dlls.First(); dll; dll = pluginManager->dlls.Next(dll)) {
|
|
cPlugin *p = dll->Plugin();
|
|
if (p && p->Service(Id, Data))
|
|
found = true;
|
|
}
|
|
}
|
|
return found;
|
|
}
|
|
|
|
void cPluginManager::StopPlugins(void)
|
|
{
|
|
for (cDll *dll = dlls.Last(); dll; dll = dlls.Prev(dll)) {
|
|
cPlugin *p = dll->Plugin();
|
|
if (p && p->started) {
|
|
isyslog("stopping plugin: %s", p->Name());
|
|
p->Stop();
|
|
p->started = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
void cPluginManager::Shutdown(bool Log)
|
|
{
|
|
cDll *dll;
|
|
while ((dll = dlls.Last()) != NULL) {
|
|
cPlugin *p = dll->Plugin();
|
|
if (p && Log)
|
|
isyslog("deleting plugin: %s", p->Name());
|
|
dlls.Del(dll);
|
|
}
|
|
}
|