mirror of
https://github.com/vdr-projects/vdr.git
synced 2025-03-01 10:50:46 +00:00
- Improved responsiveness inside CAM menus. - Added handling of the 'Close MMI' tag to avoid error log messages with CAMs that actually use it. - Now waiting at startup until all DVB devices are ready. This includes having all CAMs initialized and ready to decrypt, so that no more "channel not available" happens if VDR is started with the current channel being an encrypted one, or a timer on such a channel hits right after starting VDR. - Fixed cVideoRepacker to better handle errors in data (thanks to Reinhard Nissl). - Fixed cDvbTuner to avoid lockups on NPTL systems (thanks to Marcel Wiesweg). - Added 'Service' functions to the plugin interface (thanks to Udo Richter). See PLUGINS.html, section "Custom services" for details. - Replaced the get/put_unaligned() macros from <asm/unaligned.h> with own inline functions to avoid problems on platforms that don't provide these (thanks to David Woodhouse for his help).
427 lines
10 KiB
C
427 lines
10 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 1.14 2005/08/21 09:35:28 kls Exp $
|
|
*/
|
|
|
|
#include "plugin.h"
|
|
#include <ctype.h>
|
|
#include <dirent.h>
|
|
#include <dlfcn.h>
|
|
#include <stdlib.h>
|
|
#include <time.h>
|
|
#include "config.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()
|
|
{
|
|
I18nRegister(NULL, Name());
|
|
}
|
|
|
|
void cPlugin::SetName(const char *s)
|
|
{
|
|
name = s;
|
|
}
|
|
|
|
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)
|
|
{
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
void cPlugin::RegisterI18n(const tI18nPhrase * const Phrases)
|
|
{
|
|
I18nRegister(Phrases, Name());
|
|
}
|
|
|
|
void cPlugin::SetConfigDirectory(const char *Dir)
|
|
{
|
|
configDirectory = strdup(Dir);
|
|
}
|
|
|
|
const char *cPlugin::ConfigDirectory(const char *PluginName)
|
|
{
|
|
static char *buffer = NULL;
|
|
free(buffer);
|
|
asprintf(&buffer, "%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 !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, VDRVERSION) == 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;
|
|
char *buffer = NULL;
|
|
asprintf(&buffer, "%s/%s%s%s%s", directory, LIBVDR_PREFIX, s, SO_INDICATOR, VDRVERSION);
|
|
dlls.Add(new cDll(buffer, Args));
|
|
free(buffer);
|
|
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) {
|
|
int Language = Setup.OSDLanguage;
|
|
Setup.OSDLanguage = 0; // the i18n texts are only available _after_ Start()
|
|
isyslog("initializing plugin: %s (%s): %s", p->Name(), p->Version(), p->Description());
|
|
Setup.OSDLanguage = Language;
|
|
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) {
|
|
int Language = Setup.OSDLanguage;
|
|
Setup.OSDLanguage = 0; // the i18n texts are only available _after_ Start()
|
|
isyslog("starting plugin: %s", p->Name());
|
|
Setup.OSDLanguage = Language;
|
|
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);
|
|
}
|
|
}
|
|
|
|
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) {
|
|
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(void)
|
|
{
|
|
cDll *dll;
|
|
while ((dll = dlls.Last()) != NULL) {
|
|
cPlugin *p = dll->Plugin();
|
|
if (p)
|
|
isyslog("deleting plugin: %s", p->Name());
|
|
dlls.Del(dll);
|
|
}
|
|
}
|