Rivendellaudio/ripcd/livewire_mcastgpio.cpp
Fred Gleason ac4d9305bc 2019-06-25 Fred Gleason <fredg@paravelsystems.com>
* Adjusted to priority of 'operations' system events to use
	LOG_INFO priority.
2019-06-25 15:08:12 -04:00

504 lines
14 KiB
C++

// livewire_mcastgpio.cpp
//
// A Rivendell multicast GPIO driver for LiveWire networks.
//
// (C) Copyright 2013-2019 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 <errno.h>
#include <fcntl.h>
#include <stdlib.h>
#include <syslog.h>
#include <time.h>
#include <netinet/in.h>
#include <netinet/ip.h>
#include <rdapplication.h>
#include <rddb.h>
#include <rdescape_string.h>
#include "globals.h"
#include "livewire_mcastgpio.h"
LiveWireMcastGpio::LiveWireMcastGpio(RDMatrix *matrix,QObject *parent)
: Switcher(matrix,parent)
{
livewire_gpio_notify=NULL;
long sockopt;
struct sockaddr_in sa;
QString sql;
RDSqlQuery *q;
//
// Get Matrix Parameters
//
livewire_stationname=rda->station()->name();
livewire_matrix=matrix->matrix();
livewire_gpios=matrix->gpis();
livewire_interface_addr=matrix->ipAddress(RDMatrix::Primary);
//
// Initialize Serial Numbers
//
time_t t;
t=time(&t);
srand(t);
livewire_gpio_send_serial=rand();
//
// Timers
//
livewire_gpi_timer_mapper=new QSignalMapper(this);
connect(livewire_gpi_timer_mapper,SIGNAL(mapped(int)),
this,SLOT(gpiTimeoutData(int)));
for(unsigned i=0;i<livewire_gpios;i++) {
livewire_gpi_timers.push_back(new QTimer(this));
livewire_gpi_timer_mapper->setMapping(livewire_gpi_timers.back(),i);
connect(livewire_gpi_timers.back(),SIGNAL(timeout()),
livewire_gpi_timer_mapper,SLOT(map()));
}
livewire_gpo_out_timer_mapper=new QSignalMapper(this);
connect(livewire_gpo_out_timer_mapper,SIGNAL(mapped(int)),
this,SLOT(gpoOutTimeoutData(int)));
for(unsigned i=0;i<livewire_gpios;i++) {
livewire_gpo_out_timers.push_back(new QTimer(this));
livewire_gpo_out_timer_mapper->setMapping(livewire_gpo_out_timers.back(),i);
connect(livewire_gpo_out_timers.back(),SIGNAL(timeout()),
livewire_gpo_out_timer_mapper,SLOT(map()));
livewire_gpo_out_states.push_back(false);
}
livewire_gpo_in_timer_mapper=new QSignalMapper(this);
connect(livewire_gpo_in_timer_mapper,SIGNAL(mapped(int)),
this,SLOT(gpoInTimeoutData(int)));
for(unsigned i=0;i<livewire_gpios;i++) {
livewire_gpo_in_timers.push_back(new QTimer(this));
livewire_gpo_in_timer_mapper->setMapping(livewire_gpo_in_timers.back(),i);
connect(livewire_gpo_in_timers.back(),SIGNAL(timeout()),
livewire_gpo_in_timer_mapper,SLOT(map()));
livewire_gpo_in_states.push_back(false);
}
//
// GPIO Write Socket
//
if((livewire_gpio_write_socket=socket(PF_INET,SOCK_DGRAM,0))<0) {
rda->syslog(LOG_ERR,"unable to create GPIO write socket [%s]",
strerror(errno));
return;
}
sockopt=O_NONBLOCK;
fcntl(livewire_gpio_write_socket,F_SETFL,sockopt);
memset(&sa,0,sizeof(sa));
sa.sin_family=AF_INET;
sa.sin_port=htons(htons(RD_LIVEWIRE_GPIO_SEND_PORT));
sa.sin_addr.s_addr=
htonl(matrix->ipAddress(RDMatrix::Primary).ip4Addr());
if(bind(livewire_gpio_write_socket,(struct sockaddr *)&sa,sizeof(sa))<0) {
rda->syslog(LOG_ERR,"unable to bind GPIO write socket [%s]",
strerror(errno));
return;
}
//
// GPIO Read Socket
//
if((livewire_gpio_read_socket=socket(PF_INET,SOCK_DGRAM,0))<0) {
rda->syslog(LOG_ERR,"unable to create GPIO read socket [%s]",
strerror(errno));
return;
}
sockopt=O_NONBLOCK;
fcntl(livewire_gpio_read_socket,F_SETFL,sockopt);
memset(&sa,0,sizeof(sa));
sa.sin_family=AF_INET;
sa.sin_port=htons(RD_LIVEWIRE_GPIO_RECV_PORT);
sa.sin_addr.s_addr=htonl(INADDR_ANY);
if(bind(livewire_gpio_read_socket,(struct sockaddr *)&sa,sizeof(sa))<0) {
rda->syslog(LOG_ERR,"unable to bind GPIO socket [%s]",strerror(errno));
return;
}
livewire_gpio_notify=
new QSocketNotifier(livewire_gpio_read_socket,QSocketNotifier::Read,this);
connect(livewire_gpio_notify,SIGNAL(activated(int)),
this,SLOT(gpioActivatedData(int)));
subscribe(QHostAddress(RD_LIVEWIRE_GPIO_MCAST_ADDR));
//
// Source Table
//
sql=QString("select ")+
"SLOT,"+ // 00
"SOURCE_NUMBER,"+ // 01
"IP_ADDRESS "+ // 02
"from LIVEWIRE_GPIO_SLOTS "+
"where (STATION_NAME=\""+RDEscapeString(livewire_stationname)+"\")&&"+
QString().sprintf("(MATRIX=%d) ",livewire_matrix)+
"order by SLOT";
q=new RDSqlQuery(sql);
while(q->next()) {
if(q->value(0).toInt()<((int)livewire_gpios/RD_LIVEWIRE_GPIO_BUNDLE_SIZE)) {
livewire_source_numbers[q->value(0).toInt()]=q->value(1).toInt();
livewire_surface_addresses[q->value(0).toInt()]=
QHostAddress(q->value(2).toString());
}
}
delete q;
}
LiveWireMcastGpio::~LiveWireMcastGpio()
{
if(livewire_gpio_notify!=NULL) {
close(livewire_gpio_read_socket);
delete livewire_gpio_notify;
close(livewire_gpio_write_socket);
}
for(unsigned i=0;i<livewire_gpi_timers.size();i++) {
delete livewire_gpi_timers[i];
}
for(unsigned i=0;i<livewire_gpo_out_timers.size();i++) {
delete livewire_gpo_out_timers[i];
}
for(unsigned i=0;i<livewire_gpo_in_timers.size();i++) {
delete livewire_gpo_in_timers[i];
}
}
RDMatrix::Type LiveWireMcastGpio::type()
{
return RDMatrix::LiveWireMcastGpio;
}
unsigned LiveWireMcastGpio::gpiQuantity()
{
return livewire_gpios;
}
unsigned LiveWireMcastGpio::gpoQuantity()
{
return livewire_gpios;
}
bool LiveWireMcastGpio::primaryTtyActive()
{
return false;
}
bool LiveWireMcastGpio::secondaryTtyActive()
{
return false;
}
void LiveWireMcastGpio::processCommand(RDMacro *cmd)
{
int slot;
int line;
switch(cmd->command()) {
case RDMacro::GO:
if((cmd->argQuantity()!=5)||
(cmd->arg(2).toInt()<1)||
(cmd->arg(2).toInt()>(int)livewire_gpios)||
((cmd->arg(1).lower()!="i")&&
(cmd->arg(1).lower()!="o"))) {
cmd->acknowledge(false);
emit rmlEcho(cmd);
return;
}
if(cmd->arg(1).lower()=="i") {
slot=(cmd->arg(2).toInt()-1)/5;
line=(cmd->arg(2).toInt()-1)%5;
if(livewire_source_numbers[slot]<=0) {
cmd->acknowledge(false);
emit rmlEcho(cmd);
return;
}
if(cmd->arg(4).toInt()>0) {
livewire_gpo_in_timers[cmd->arg(2).toInt()-1]->
start(cmd->arg(4).toInt(),true);
}
ProcessGpoIn(livewire_source_numbers[slot],line,cmd->arg(3).toInt());
livewire_gpo_in_states[cmd->arg(2).toInt()-1]=cmd->arg(3).toInt();
}
if(cmd->arg(1).lower()=="o") {
slot=(cmd->arg(2).toInt()-1)/5;
line=(cmd->arg(2).toInt()-1)%5;
if(livewire_source_numbers[slot]<=0) {
cmd->acknowledge(false);
emit rmlEcho(cmd);
return;
}
if(cmd->arg(4).toInt()>0) {
livewire_gpo_out_timers[cmd->arg(2).toInt()-1]->
start(cmd->arg(4).toInt(),true);
}
ProcessGpoOut(livewire_source_numbers[slot],line,cmd->arg(3).toInt());
livewire_gpo_out_states[cmd->arg(2).toInt()-1]=cmd->arg(3).toInt();
}
cmd->acknowledge(true);
emit rmlEcho(cmd);
break;
default:
cmd->acknowledge(false);
emit rmlEcho(cmd);
break;
}
}
void LiveWireMcastGpio::gpioActivatedData(int sock)
{
int n;
char data[1500];
struct sockaddr_in sa;
socklen_t sa_len=sizeof(sa);
uint32_t serial;
while((n=recvfrom(livewire_gpio_read_socket,data,1500,MSG_DONTWAIT,
(struct sockaddr *)(&sa),&sa_len))>0) {
serial=((0xFF&data[4])<<24)+((0xFF&data[5])<<16)+((0xFF&data[6])<<8)+
(0xFF&data[7]);
if((livewire_gpio_recv_serials[sa.sin_addr.s_addr]!=serial)&&
(livewire_gpio_recv_serials[sa.sin_addr.s_addr]!=(serial-1))) {
livewire_gpio_recv_serials[sa.sin_addr.s_addr]=serial;
ProcessGpi(QHostAddress(ntohl(sa.sin_addr.s_addr)),
((0xFF&data[23])<<8)+(0xFF&data[24]),0x08-(0xff&data[25]),
(data[27]&0x40)!=0,(data[27]&0x0A)!=0);
}
}
}
void LiveWireMcastGpio::gpiTimeoutData(int gpi)
{
emit gpiChanged(livewire_matrix,gpi,false);
}
void LiveWireMcastGpio::gpoInTimeoutData(int gpo)
{
int slot=gpo/5;
int line=gpo%5;
if(livewire_source_numbers[slot]>0) {
ProcessGpoIn(livewire_source_numbers[slot],line,!livewire_gpo_in_states[gpo]);
livewire_gpo_in_states[gpo]=!livewire_gpo_in_states[gpo];
}
}
void LiveWireMcastGpio::gpoOutTimeoutData(int gpo)
{
int slot=gpo/5;
int line=gpo%5;
if(livewire_source_numbers[slot]>0) {
ProcessGpoOut(livewire_source_numbers[slot],line,!livewire_gpo_out_states[gpo]);
livewire_gpo_out_states[gpo]=!livewire_gpo_out_states[gpo];
}
}
void LiveWireMcastGpio::ProcessGpi(const QHostAddress &src_addr,int chan,
unsigned line,bool state,bool pulse)
{
for(std::map<int,int>::const_iterator it=livewire_source_numbers.begin();
it!=livewire_source_numbers.end();it++) {
if((it->second==chan)&&
((livewire_surface_addresses[it->first].isNull())||
(livewire_surface_addresses[it->first]==src_addr))) {
emit gpiChanged(livewire_matrix,5*it->first+line,state);
if(pulse) {
livewire_gpi_timers[5*it->first+line]->
start(RD_LIVEWIRE_GPIO_PULSE_WIDTH,true);
}
}
}
}
void LiveWireMcastGpio::ProcessGpoIn(int chan,unsigned line,bool state)
{
//
// Destination Address
//
struct sockaddr_in sa;
memset(&sa,0,sizeof(sa));
sa.sin_family=AF_INET;
sa.sin_port=htons(RD_LIVEWIRE_GPIO_RECV_PORT);
sa.sin_addr.s_addr=
htonl(QHostAddress(RD_LIVEWIRE_GPIO_MCAST_ADDR).toIPv4Address());
uint8_t data[60]={0x03,0x00,0x02,0x07,0x36,0x0B,0x97,0xA9,
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
'W','R','N','I',0x00,0x04,0x00,0x03,0xF6,
0x05,0x07,0xCA,0xFF,0xFF,0xFF,0xFD,0x07,
0x01,0xFF,0xFF,0xFF,0xFC,0x09,0x00,0x02,
0x15,0x07,0x00,0x12,0x00,0x8F,0xFF,0xFF,
0xFF,0xFF,0x09,0x00,0x02,0x15,0x07,0x00,
0x00,0x00,0x8F};
data[4]=0xFF&(livewire_gpio_send_serial>>24);
data[5]=0xFF&(livewire_gpio_send_serial>>16);
data[6]=0xFF&(livewire_gpio_send_serial>>8);
data[7]=0xFF&livewire_gpio_send_serial;
data[23]=0xFF&(chan>>8);
data[24]=0xFF&chan;
data[25]=0x08-(0x07&line);
if(state) {
data[27]|=0x40;
}
else {
data[27]&=~0x40;
}
data[27]&=~0x0A; // No pulse
sendto(livewire_gpio_write_socket,data,28,MSG_DONTWAIT,
(struct sockaddr *)(&sa),sizeof(sa));
livewire_gpio_send_serial++;
data[4]=0xFF&(livewire_gpio_send_serial>>24);
data[5]=0xFF&(livewire_gpio_send_serial>>16);
data[6]=0xFF&(livewire_gpio_send_serial>>8);
data[7]=0xFF&livewire_gpio_send_serial;
sendto(livewire_gpio_write_socket,data,60,MSG_DONTWAIT,
(struct sockaddr *)(&sa),sizeof(sa));
livewire_gpio_send_serial+=2;
}
void LiveWireMcastGpio::ProcessGpoOut(int chan,unsigned line,bool state)
{
//
// Destination Address
//
struct sockaddr_in sa;
memset(&sa,0,sizeof(sa));
sa.sin_family=AF_INET;
sa.sin_port=htons(RD_LIVEWIRE_GPIO_SEND_PORT);
sa.sin_addr.s_addr=
htonl(QHostAddress(RD_LIVEWIRE_GPIO_MCAST_ADDR).toIPv4Address());
/*
* FIXME: Sending this to the mcast addr causes ALL instances
* of the source to switch. How does one specify a particular
* surface? Sending to the surface addr does not work!
*
for(std::map<int,int>::const_iterator it=livewire_source_numbers.begin();
it!=livewire_source_numbers.end();it++) {
if((it->second==chan)&&(!livewire_surface_addresses[it->first].isNull())) {
sa.sin_addr.s_addr=
htonl(livewire_surface_addresses[it->first].toIPv4Address());
}
}
rda->syslog(LOG_DEBUG,"using %s",
(const char *)QHostAddress(ntohl(sa.sin_addr.s_addr)).toString());
*/
uint8_t data[28]={0x03,0x00,0x02,0x07,0xC6,0x04,0x55,0x1E,
0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
'I','N','D','I',0x00,0x01,0x00,0x00,
0x00,0x00,0x07,0x00};
data[4]=0xFF&(livewire_gpio_send_serial>>24);
data[5]=0xFF&(livewire_gpio_send_serial>>16);
data[6]=0xFF&(livewire_gpio_send_serial>>8);
data[7]=0xFF&livewire_gpio_send_serial;
data[23]=0xFF&(chan>>8);
data[24]=0xFF&chan;
data[25]=0xFF&(0x0D-line);
data[27]=0xFF&state;
sendto(livewire_gpio_write_socket,data,28,MSG_DONTWAIT,
(struct sockaddr *)(&sa),sizeof(sa));
livewire_gpio_send_serial++;
data[4]=0xFF&(livewire_gpio_send_serial>>24);
data[5]=0xFF&(livewire_gpio_send_serial>>16);
data[6]=0xFF&(livewire_gpio_send_serial>>8);
data[7]=0xFF&livewire_gpio_send_serial;
sendto(livewire_gpio_write_socket,data,28,MSG_DONTWAIT,
(struct sockaddr *)(&sa),sizeof(sa));
livewire_gpio_send_serial+=2;
}
QString LiveWireMcastGpio::AddressString(uint32_t addr) const
{
return QString().sprintf("%d.%d.%d.%d",
0xFF&addr,
0xFF&(addr>>8),
0xFF&(addr>>16),
0xFF&(addr>>24));
}
void LiveWireMcastGpio::subscribe(const QHostAddress &addr) const
{
subscribe(htonl(addr.toIPv4Address()));
}
void LiveWireMcastGpio::subscribe(const uint32_t addr) const
{
struct ip_mreqn mreq;
memset(&mreq,0,sizeof(mreq));
mreq.imr_multiaddr.s_addr=addr;
mreq.imr_address.s_addr=htonl(livewire_interface_addr.toIPv4Address());
mreq.imr_ifindex=0;
if(setsockopt(livewire_gpio_read_socket,SOL_IP,IP_ADD_MEMBERSHIP,&mreq,
sizeof(mreq))!=0) {
if(errno!=EADDRINUSE) {
rda->syslog(LOG_WARNING,"unable to subscribe to %s on %s [%s]",
(const char *)AddressString(addr),
(const char *)livewire_interface_addr.toString(),
strerror(errno));
}
}
}
void LiveWireMcastGpio::unsubscribe(const QHostAddress &addr) const
{
unsubscribe(htonl(addr.toIPv4Address()));
}
void LiveWireMcastGpio::unsubscribe(const uint32_t addr) const
{
struct ip_mreqn mreq;
memset(&mreq,0,sizeof(mreq));
mreq.imr_multiaddr.s_addr=htonl(addr);
mreq.imr_address.s_addr=htonl(livewire_interface_addr.toIPv4Address());
mreq.imr_ifindex=0;
if(setsockopt(livewire_gpio_read_socket,SOL_IP,IP_DROP_MEMBERSHIP,&mreq,
sizeof(mreq))!=0) {
if(errno!=ENODEV) {
rda->syslog(LOG_WARNING,"unable to unsubscribe from %s on %s [%s]",
(const char *)AddressString(addr),
(const char *)livewire_interface_addr.toString(),
strerror(errno));
}
}
}