Rivendellaudio/lib/rdrlmhost.cpp
Fred Gleason 796e1bebc3 2018-08-07 Fred Gleason <fredg@paravelsystems.com>
* Removed the 'pam_rd' plug-in.
	* Removed rdchunk(1).
	* Upgraded Qt3 to Qt4.
2018-08-07 10:09:49 -04:00

543 lines
15 KiB
C++

// rdrlmhost.cpp
//
// A container class for a Rivendell Loadable Module host.
//
// (C) Copyright 2008,2016-2018 Fred Gleason <fredg@paravelsystems.com>
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License version 2 as
// published by the Free Software Foundation.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public
// License along with this program; if not, write to the Free Software
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
//
#include <dlfcn.h>
#include <iostream>
#include <qsignalmapper.h>
#include "rdapplication.h"
#include "rdconf.h"
#include "rddatedecode.h"
#include "rdnownext.h"
#include "rdprofile.h"
#include "rdrlmhost.h"
#include "rdsvc.h"
//#include "globals.h"
RDRLMHost::RDRLMHost(const QString &path,const QString &arg,
Q3SocketDevice *udp_socket,QObject *parent)
: QObject(parent)
{
plugin_path=RDDateDecode(path,QDate::currentDate(),rda->station(),rda->config());
plugin_arg=RDDateDecode(arg,QDate::currentDate(),rda->station(),rda->config());
plugin_udp_socket=udp_socket;
plugin_handle=NULL;
plugin_start_sym=NULL;
plugin_free_sym=NULL;
plugin_pad_data_sent_sym=NULL;
plugin_timer_expired_sym=NULL;
plugin_serial_data_received_sym=NULL;
//
// Utility Timers
//
QSignalMapper *mapper=new QSignalMapper(this);
connect(mapper,SIGNAL(mapped(int)),this,SLOT(timerData(int)));
for(int i=0;i<RLM_MAX_TIMERS;i++) {
plugin_callback_timers[i]=new QTimer(this);
mapper->setMapping(plugin_callback_timers[i],i);
connect(plugin_callback_timers[i],SIGNAL(timeout()),mapper,SLOT(map()));
}
}
RDRLMHost::~RDRLMHost()
{
}
QString RDRLMHost::pluginPath() const
{
return plugin_path;
}
QString RDRLMHost::pluginArg() const
{
return plugin_arg;
}
void RDRLMHost::sendEvent(const QString &svcname,const QString &logname,
int lognum,RDLogLine **loglines,bool onair,
RDAirPlayConf::OpMode mode)
{
if(plugin_pad_data_sent_sym!=NULL) {
QDateTime now_dt(QDate::currentDate(),QTime::currentTime());
struct rlm_svc *svc=new struct rlm_svc;
struct rlm_log *log=new struct rlm_log;
struct rlm_pad *now=new struct rlm_pad;
struct rlm_pad *next=new struct rlm_pad;
memset(svc,0,sizeof(struct rlm_svc));
RDSvc *service=new RDSvc(svcname,rda->station(),rda->config());
if(!svcname.isEmpty()) {
sprintf(svc->svc_name,"%s",(const char *)svcname.left(255));
if(!service->programCode().isEmpty()) {
sprintf(svc->svc_pgmcode,"%s",(const char *)service->programCode());
}
else {
svc->svc_pgmcode[0]=0;
}
}
else {
svc->svc_name[0]=0;
svc->svc_pgmcode[0]=0;
}
delete service;
memset(log,0,sizeof(struct rlm_log));
if(!logname.isEmpty()) {
sprintf(log->log_name,"%s",(const char *)logname.left(64));
}
else {
log->log_name[0]=0;
}
log->log_mach=lognum;
log->log_onair=onair;
log->log_mode=mode;
RDRLMHost::loadMetadata(loglines[0],now,now_dt);
RDRLMHost::loadMetadata(loglines[1],next);
plugin_pad_data_sent_sym(this,svc,log,now,next);
delete next;
delete now;
delete log;
delete svc;
}
}
bool RDRLMHost::load()
{
QString basename=RDGetBasePart(plugin_path);
basename=basename.left(basename.findRev("."));
if((plugin_handle=dlopen(plugin_path,RTLD_LAZY))==NULL) {
return false;
}
*(void **)(&plugin_start_sym)=dlsym(plugin_handle,basename+"_RLMStart");
*(void **)(&plugin_free_sym)=dlsym(plugin_handle,basename+"_RLMFree");
*(void **)(&plugin_pad_data_sent_sym)=
dlsym(plugin_handle,basename+"_RLMPadDataSent");
*(void **)(&plugin_timer_expired_sym)=
dlsym(plugin_handle,basename+"_RLMTimerExpired");
*(void **)(&plugin_serial_data_received_sym)=
dlsym(plugin_handle,basename+"_RLMSerialDataReceived");
if(plugin_start_sym!=NULL) {
plugin_start_sym(this,plugin_arg);
}
return true;
}
void RDRLMHost::unload()
{
if(plugin_free_sym!=NULL) {
plugin_free_sym(this);
}
}
void RDRLMHost::loadMetadata(const RDLogLine *logline,struct rlm_pad *pad,
const QDateTime &start_datetime)
{
QDateTime now(QDate::currentDate(),QTime::currentTime());
if(pad==NULL) {
return;
}
memset(pad,0,sizeof(struct rlm_pad));
if(logline==NULL) {
return;
}
if(logline!=NULL) {
pad->rlm_cartnum=logline->cartNumber();
switch(logline->cartType()) {
case RDCart::Audio:
pad->rlm_len=logline->effectiveLength();
break;
case RDCart::Macro:
if((logline->eventLength()>=0)&&logline->useEventLength()) {
pad->rlm_len=logline->eventLength();
}
else {
pad->rlm_len=logline->effectiveLength();
}
break;
case RDCart::All:
break;
}
pad->rlm_carttype=logline->cartType();
if(!logline->year().isNull()) {
snprintf(pad->rlm_year,5,"%s",
(const char *)logline->year().toString("yyyy"));
}
if(!logline->groupName().isEmpty()) {
snprintf(pad->rlm_group,11,"%s",
(const char *)logline->groupName().utf8());
}
if(!logline->title().isEmpty()) {
snprintf(pad->rlm_title,256,"%s",(const char *)logline->title().utf8());
}
if(!logline->artist().isEmpty()) {
snprintf(pad->rlm_artist,256,"%s",(const char *)logline->artist().utf8());
}
if(!logline->label().isEmpty()) {
snprintf(pad->rlm_label,65,"%s",(const char *)logline->label().utf8());
}
if(!logline->client().isEmpty()) {
snprintf(pad->rlm_client,65,"%s",(const char *)logline->client().utf8());
}
if(!logline->agency().isEmpty()) {
snprintf(pad->rlm_agency,65,"%s",(const char *)logline->agency().utf8());
}
if(!logline->composer().isEmpty()) {
snprintf(pad->rlm_comp,65,"%s",(const char *)logline->composer().utf8());
}
if(!logline->publisher().isEmpty()) {
snprintf(pad->rlm_pub,65,"%s",(const char *)logline->publisher().utf8());
}
if(!logline->userDefined().isEmpty()) {
snprintf(pad->rlm_userdef,256,"%s",
(const char *)logline->userDefined().utf8());
}
if(!logline->outcue().isEmpty()) {
snprintf(pad->rlm_outcue,65,"%s",(const char *)logline->outcue().utf8());
}
if(!logline->description().isEmpty()) {
snprintf(pad->rlm_description,65,"%s",
(const char *)logline->description().utf8());
}
if(!logline->conductor().isEmpty()) {
snprintf(pad->rlm_conductor,65,"%s",
(const char *)logline->conductor().utf8());
}
if(!logline->songId().isEmpty()) {
snprintf(pad->rlm_song_id,33,"%s",(const char *)logline->songId().utf8());
}
if(!logline->album().isEmpty()) {
snprintf(pad->rlm_album,256,"%s",(const char *)logline->album().utf8());
}
if(!logline->isrc().isEmpty()) {
strncpy(pad->rlm_isrc,(const char *)logline->isrc().utf8().left(12),12);
}
if(!logline->isci().isEmpty()) {
strncpy(pad->rlm_isci,(const char *)logline->isci().utf8().left(32),32);
}
if(!logline->extData().isEmpty()) {
snprintf(pad->rlm_ext_data,32,"%s",(const char *)logline->extData());
}
if(!logline->extEventId().isEmpty()) {
snprintf(pad->rlm_ext_eventid,32,"%s",
(const char *)logline->extEventId());
}
if(!logline->extAnncType().isEmpty()) {
snprintf(pad->rlm_ext_annctype,32,"%s",
(const char *)logline->extAnncType());
}
if(start_datetime.isValid()) {
pad->rlm_start_msec=start_datetime.time().msec();
pad->rlm_start_sec=start_datetime.time().second();
pad->rlm_start_min=start_datetime.time().minute();
pad->rlm_start_hour=start_datetime.time().hour();
pad->rlm_start_day=start_datetime.date().day();
pad->rlm_start_mon=start_datetime.date().month();
pad->rlm_start_year=start_datetime.date().year();
}
else {
QTime start_time=logline->startTime(RDLogLine::Predicted);
if(start_time.isNull()) {
start_time=logline->startTime(RDLogLine::Imported);
}
if(!start_time.isNull()) {
if(start_time<now.time()) { // Crossing midnight
now=now.addDays(1);
}
}
pad->rlm_start_msec=start_time.msec();
pad->rlm_start_sec=start_time.second();
pad->rlm_start_min=start_time.minute();
pad->rlm_start_hour=start_time.hour();
pad->rlm_start_day=now.date().day();
pad->rlm_start_mon=now.date().month();
pad->rlm_start_year=now.date().year();
}
}
}
void RDRLMHost::saveMetadata(const struct rlm_pad *pad,RDLogLine *logline)
{
if(logline==NULL) {
return;
}
logline->clear();
if(pad==NULL) {
return;
}
logline->setCartNumber(pad->rlm_cartnum);
logline->setForcedLength(pad->rlm_len);
logline->setYear(QDate(QString(pad->rlm_year).toInt(),1,1));
logline->setGroupName(pad->rlm_group);
logline->setTitle(pad->rlm_title);
logline->setArtist(pad->rlm_artist);
logline->setLabel(pad->rlm_label);
logline->setClient(pad->rlm_client);
logline->setAgency(pad->rlm_agency);
logline->setComposer(pad->rlm_comp);
logline->setPublisher(pad->rlm_pub);
logline->setUserDefined(pad->rlm_userdef);
logline->setOutcue(pad->rlm_outcue);
logline->setDescription(pad->rlm_description);
logline->setAlbum(pad->rlm_album);
logline->setIsrc(QString::fromAscii(pad->rlm_isrc,12));
logline->setIsci(QString::fromAscii(pad->rlm_isci,32));
if((pad->rlm_start_year>0)&&(pad->rlm_start_mon>0)&&(pad->rlm_start_day)) {
logline->setStartDatetime(QDateTime(QDate(pad->rlm_start_year,
pad->rlm_start_mon,
pad->rlm_start_day),
QTime(pad->rlm_start_hour,
pad->rlm_start_min,
pad->rlm_start_sec,
pad->rlm_start_msec)));
}
else {
logline->setStartDatetime(QDateTime());
}
}
void RDRLMHost::timerData(int timernum)
{
if(plugin_timer_expired_sym!=NULL) {
plugin_timer_expired_sym(this,timernum);
}
}
void RDRLMHost::ttyReceiveReadyData(int fd)
{
char data[1024];
int n;
for(unsigned i=0;i<plugin_tty_devices.size();i++) {
if(plugin_tty_devices[i]->fileDescriptor()==fd) {
while((n=plugin_tty_devices[i]->readBlock(data,1024))>0) {
if(plugin_serial_data_received_sym!=NULL) {
plugin_serial_data_received_sym(this,i,data,n);
}
}
return;
}
}
fprintf(stderr,"unknown tty descriptor: %d\n",fd);
}
//
// RLM Utility Functions
//
void RLMSendUdp(void *ptr,const char *ipaddr,uint16_t port,
const char *data,int len)
{
RDRLMHost *host=(RDRLMHost *)ptr;
QHostAddress addr;
addr.setAddress(ipaddr);
if(!addr.isNull()) {
host->plugin_udp_socket->writeBlock(data,len,addr,port);
}
}
int RLMOpenSerial(void *ptr,const char *port,int speed,int parity,
int word_length)
{
RDRLMHost *host=(RDRLMHost *)ptr;
host->plugin_tty_devices.push_back(new RDTTYDevice());
host->plugin_tty_devices.back()->setName(port);
host->plugin_tty_devices.back()->setSpeed(speed);
host->plugin_tty_devices.back()->setParity((RDTTYDevice::Parity)parity);
host->plugin_tty_devices.back()->setWordLength(word_length);
if(host->plugin_tty_devices.back()->open(QIODevice::ReadWrite)) {
host->plugin_tty_notifiers.
push_back(new QSocketNotifier(host->plugin_tty_devices.back()->fileDescriptor(),
QSocketNotifier::Read));
host->connect(host->plugin_tty_notifiers.back(),SIGNAL(activated(int)),
host,SLOT(ttyReceiveReadyData(int)));
return (int)host->plugin_tty_devices.size()-1;
}
return -1;
}
void RLMSendSerial(void *ptr,int handle,const char *data,int len)
{
RDRLMHost *host=(RDRLMHost *)ptr;
if((handle<0)||(handle>=(int)host->plugin_tty_devices.size())) {
return;
}
host->plugin_tty_devices[handle]->writeBlock(data,len);
}
void RLMCloseSerial(void *ptr,int handle)
{
RDRLMHost *host=(RDRLMHost *)ptr;
//
// FIXME: We really ought to take out the trash here!
//
host->plugin_tty_devices[handle]->close();
delete host->plugin_tty_devices[handle];
host->plugin_tty_devices[handle]=NULL;
}
const char *RLMDateTime(void *ptr,int offset_msecs,const char *format)
{
RDRLMHost *host=(RDRLMHost *)ptr;
QDateTime datetime=QDateTime(QDate::currentDate(),QTime::currentTime().
addMSecs(offset_msecs));
strncpy(host->plugin_value_string,datetime.toString(format),1024);
return host->plugin_value_string;
}
const char *RLMResolveNowNextEncoded(void *ptr,const struct rlm_pad *now,
const struct rlm_pad *next,
const char *format,int encoding)
{
RDRLMHost *host=(RDRLMHost *)ptr;
RDLogLine *loglines[2];
QString str=format;
loglines[0]=new RDLogLine();
loglines[1]=new RDLogLine();
RDRLMHost::saveMetadata(now,loglines[0]);
RDRLMHost::saveMetadata(next,loglines[1]);
RDResolveNowNext(&str,loglines,encoding);
strncpy(host->plugin_value_string,str,1024);
delete loglines[1];
delete loglines[0];
return host->plugin_value_string;
}
const char *RLMResolveNowNext(void *ptr,const struct rlm_pad *now,
const struct rlm_pad *next,const char *format)
{
return RLMResolveNowNextEncoded(ptr,now,next,format,RLM_ENCODE_NONE);
}
void RLMLog(void *ptr,int prio,const char *msg)
{
rda->config()->log("log machine",(RDConfig::LogPriority)prio,msg);
}
void RLMStartTimer(void *ptr,int timernum,int msecs,int mode)
{
RDRLMHost *host=(RDRLMHost *)ptr;
if((timernum<0)||(timernum>=RLM_MAX_TIMERS)) {
return;
}
if(host->plugin_callback_timers[timernum]->isActive()) {
host->plugin_callback_timers[timernum]->stop();
}
host->plugin_callback_timers[timernum]->start(msecs,mode);
}
void RLMStopTimer(void *ptr,int timernum)
{
RDRLMHost *host=(RDRLMHost *)ptr;
if((timernum<0)||(timernum>=RLM_MAX_TIMERS)) {
return;
}
if(host->plugin_callback_timers[timernum]->isActive()) {
host->plugin_callback_timers[timernum]->stop();
}
}
int RLMGetIntegerValue(void *ptr,const char *filename,const char *section,
const char *label,int default_value)
{
RDProfile *p=new RDProfile();
p->setSource(filename);
int r=p->intValue(section,label,default_value);
delete p;
return r;
}
int RLMGetHexValue(void *ptr,const char *filename,const char *section,
const char *label,int default_value)
{
RDProfile *p=new RDProfile();
p->setSource(filename);
int r=p->hexValue(section,label,default_value);
delete p;
return r;
}
int RLMGetBooleanValue(void *ptr,const char *filename,const char *section,
const char *label,int default_value)
{
RDProfile *p=new RDProfile();
p->setSource(filename);
bool r=p->boolValue(section,label,default_value);
delete p;
return (int)r;
}
const char *RLMGetStringValue(void *ptr,const char *filename,
const char *section,const char *label,
const char *default_value)
{
RDRLMHost *host=(RDRLMHost *)ptr;
RDProfile *p=new RDProfile();
p->setSource(filename);
strncpy(host->plugin_value_string,
p->stringValue(section,label,default_value),1024);
delete p;
return host->plugin_value_string;
}
const char *RLMDateTimeDecode(void *ptr, const char *format,
const char *svc_name)
{
RDRLMHost *host=(RDRLMHost *)ptr;
strncpy(host->plugin_value_string,
RDDateTimeDecode(format,QDateTime::currentDateTime(),
rda->station(),rda->config(),
svc_name),1024);
return host->plugin_value_string;
}