mirror of
https://github.com/ElvishArtisan/rivendell.git
synced 2025-05-19 06:32:34 +02:00
* Fixed a bug in 'RDCatchEvent::read()' that caused a segfault when processing a 'Reload Decks' operation. Signed-off-by: Fred Gleason <fredg@paravelsystems.com>
537 lines
11 KiB
C++
537 lines
11 KiB
C++
// rdcatchevent.cpp
|
|
//
|
|
// A container class for a Rivendell Catch Event message.
|
|
//
|
|
// (C) Copyright 2022-2023 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 <QStringList>
|
|
|
|
#include "rdcatchevent.h"
|
|
#include "rdapplication.h"
|
|
|
|
RDCatchMeterLevel::RDCatchMeterLevel(int deck,int16_t *lvls)
|
|
{
|
|
d_deck_channel=deck;
|
|
for(int i=0;i<RDCatchMeterLevel::LastChannel;i++) {
|
|
d_levels[i]=-lvls[i];
|
|
}
|
|
}
|
|
|
|
|
|
unsigned RDCatchMeterLevel::deckChannel() const
|
|
{
|
|
return d_deck_channel;
|
|
}
|
|
|
|
|
|
int16_t RDCatchMeterLevel::level(Channel chan) const
|
|
{
|
|
return -d_levels[chan];
|
|
}
|
|
|
|
|
|
QString RDCatchMeterLevel::dump() const
|
|
{
|
|
QString lvls;
|
|
QString ret;
|
|
|
|
for(int j=0;j<RDCatchMeterLevel::LastChannel;j++) {
|
|
lvls+=QString::asprintf(":%04X",d_levels[j]);
|
|
}
|
|
ret+=QString::asprintf("%u",d_deck_channel)+lvls;
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
|
|
|
|
RDCatchEvent::RDCatchEvent(RDDeck::Status status)
|
|
{
|
|
clear();
|
|
d_operation=RDCatchEvent::DeckEventProcessedOp;
|
|
d_deck_status=status;
|
|
}
|
|
|
|
|
|
RDCatchEvent::RDCatchEvent()
|
|
{
|
|
clear();
|
|
}
|
|
|
|
|
|
RDCatchEvent::Operation RDCatchEvent::operation() const
|
|
{
|
|
return d_operation;
|
|
}
|
|
|
|
|
|
void RDCatchEvent::setOperation(RDCatchEvent::Operation op)
|
|
{
|
|
d_operation=op;
|
|
}
|
|
|
|
|
|
int RDCatchEvent::eventNumber() const
|
|
{
|
|
return d_event_number;
|
|
}
|
|
|
|
|
|
void RDCatchEvent::setEventNumber(int num)
|
|
{
|
|
d_event_number=num;
|
|
}
|
|
|
|
|
|
QString RDCatchEvent::hostName() const
|
|
{
|
|
return d_host_name;
|
|
}
|
|
|
|
|
|
void RDCatchEvent::setHostName(const QString &str)
|
|
{
|
|
d_host_name=str;
|
|
}
|
|
|
|
|
|
QString RDCatchEvent::targetHostName() const
|
|
{
|
|
return d_target_host_name;
|
|
}
|
|
|
|
|
|
void RDCatchEvent::setTargetHostName(const QString &str)
|
|
{
|
|
d_target_host_name=str;
|
|
}
|
|
|
|
|
|
unsigned RDCatchEvent::eventId() const
|
|
{
|
|
return d_event_id;
|
|
}
|
|
|
|
|
|
void RDCatchEvent::setEventId(unsigned id)
|
|
{
|
|
d_event_id=id;
|
|
}
|
|
|
|
|
|
unsigned RDCatchEvent::cartNumber() const
|
|
{
|
|
return d_cart_number;
|
|
}
|
|
|
|
|
|
void RDCatchEvent::setCartNumber(unsigned cartnum)
|
|
{
|
|
d_cart_number=cartnum;
|
|
}
|
|
|
|
|
|
int RDCatchEvent::cutNumber() const
|
|
{
|
|
return d_cut_number;
|
|
}
|
|
|
|
|
|
void RDCatchEvent::setCutNumber(int cutnum)
|
|
{
|
|
d_cut_number=cutnum;
|
|
}
|
|
|
|
|
|
unsigned RDCatchEvent::deckChannel() const
|
|
{
|
|
return d_deck_channel;
|
|
}
|
|
|
|
|
|
void RDCatchEvent::setDeckChannel(unsigned chan)
|
|
{
|
|
d_deck_channel=chan;
|
|
}
|
|
|
|
|
|
RDDeck::Status RDCatchEvent::deckStatus() const
|
|
{
|
|
return d_deck_status;
|
|
}
|
|
|
|
|
|
void RDCatchEvent::setDeckStatus(RDDeck::Status status)
|
|
{
|
|
d_deck_status=status;
|
|
}
|
|
|
|
|
|
bool RDCatchEvent::inputMonitorActive() const
|
|
{
|
|
return d_input_monitor_active;
|
|
}
|
|
|
|
|
|
void RDCatchEvent::setInputMonitorActive(bool state)
|
|
{
|
|
d_input_monitor_active=state;
|
|
}
|
|
|
|
|
|
QList<RDCatchMeterLevel> RDCatchEvent::meterLevels() const
|
|
{
|
|
return d_meter_levels;
|
|
}
|
|
|
|
|
|
void RDCatchEvent::setMeterLevels(const QList<RDCatchMeterLevel> &lvls)
|
|
{
|
|
d_meter_levels=lvls;
|
|
}
|
|
|
|
|
|
bool RDCatchEvent::read(const QString &str)
|
|
{
|
|
// printf("RDCatchEvent::read(\"%s\")\n",str.toUtf8().constData());
|
|
|
|
RDCatchEvent::Operation op=RDCatchEvent::NullOp;
|
|
QStringList f0=str.split(" ");
|
|
|
|
unsigned chan=0;
|
|
unsigned num=0;
|
|
RDDeck::Status status=RDDeck::Offline;
|
|
int state=0;
|
|
bool ok=false;
|
|
|
|
clear();
|
|
|
|
//
|
|
// Common Fields
|
|
//
|
|
if((f0.size()<3)||(f0.at(0)!="CATCH")) {
|
|
return false;
|
|
}
|
|
op=(RDCatchEvent::Operation)f0.at(2).toUInt(&ok);
|
|
if(!ok) {
|
|
return false;
|
|
}
|
|
|
|
//
|
|
// Operation-specific Fields
|
|
//
|
|
switch(op) {
|
|
case RDCatchEvent::DeckEventProcessedOp:
|
|
if(f0.size()!=5) {
|
|
return false;
|
|
}
|
|
chan=f0.at(3).toUInt(&ok);
|
|
if(!ok) {
|
|
return false;
|
|
}
|
|
num=f0.at(4).toUInt(&ok);
|
|
if(ok) {
|
|
d_operation=op;
|
|
d_host_name=f0.at(1);
|
|
d_deck_channel=chan;
|
|
d_event_number=num;
|
|
return true;
|
|
}
|
|
break;
|
|
|
|
case RDCatchEvent::SendMeterLevelsOp:
|
|
for(int i=2;i<f0.size();i++) {
|
|
QStringList f1=f0.at(i).split(":",QString::KeepEmptyParts);
|
|
if(f1.size()==(1+RDCatchMeterLevel::LastChannel)) {
|
|
chan=f1.at(0).toUInt(&ok);
|
|
if(chan>=255) {
|
|
d_meter_levels.clear();
|
|
return false;
|
|
}
|
|
int16_t lvls[RDCatchMeterLevel::LastChannel];
|
|
for(int j=0;j<RDCatchMeterLevel::LastChannel;j++) {
|
|
int lvl=f1.at(1+j).toInt(&ok,16);
|
|
if((!ok)||(lvl>0xFFFF)) {
|
|
d_meter_levels.clear();
|
|
return false;
|
|
}
|
|
lvls[j]=-(int16_t)lvl;
|
|
}
|
|
d_meter_levels.push_back(RDCatchMeterLevel(chan,lvls));
|
|
}
|
|
}
|
|
d_operation=op;
|
|
d_host_name=f0.at(1);
|
|
return true;
|
|
break;
|
|
|
|
case RDCatchEvent::DeckStatusQueryOp:
|
|
if(f0.size()!=3) {
|
|
return false;
|
|
}
|
|
d_operation=op;
|
|
d_host_name=f0.at(1);
|
|
return true;
|
|
break;
|
|
|
|
case RDCatchEvent::DeckStatusResponseOp:
|
|
if(f0.size()!=8) {
|
|
return false;
|
|
}
|
|
chan=f0.at(3).toUInt(&ok);
|
|
if(ok&&(chan<255)) {
|
|
status=(RDDeck::Status)f0.at(4).toUInt(&ok);
|
|
if(ok&&(status<RDDeck::LastStatus)) {
|
|
unsigned id=f0.at(5).toUInt(&ok);
|
|
if(ok) {
|
|
unsigned cartnum=f0.at(6).toUInt(&ok);
|
|
if(ok&&(cartnum<=RD_MAX_CART_NUMBER)) {
|
|
int cutnum=f0.at(7).toInt(&ok);
|
|
if(ok&&(cutnum>=0)&&(cutnum<=RD_MAX_CUT_NUMBER)) {
|
|
d_operation=op;
|
|
d_host_name=f0.at(1);
|
|
d_deck_channel=chan;
|
|
d_deck_status=status;
|
|
d_event_id=id;
|
|
d_cart_number=cartnum;
|
|
d_cut_number=cutnum;
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
case RDCatchEvent::StopDeckOp:
|
|
if(f0.size()!=5) {
|
|
return false;
|
|
}
|
|
chan=f0.at(4).toInt(&ok);
|
|
if(ok&&(chan<255)) {
|
|
d_operation=op;
|
|
d_host_name=f0.at(1);
|
|
d_target_host_name=f0.at(3);
|
|
d_deck_channel=chan;
|
|
return true;
|
|
}
|
|
break;
|
|
|
|
case RDCatchEvent::SetInputMonitorOp:
|
|
if(f0.size()!=6) {
|
|
return false;
|
|
}
|
|
chan=f0.at(4).toInt(&ok);
|
|
if(ok&&(chan<255)) {
|
|
state=f0.at(5).toUInt(&ok);
|
|
if((state==0)||(state==1)) {
|
|
d_operation=op;
|
|
d_host_name=f0.at(1);
|
|
d_target_host_name=f0.at(3);
|
|
d_deck_channel=chan;
|
|
d_input_monitor_active=(state==1);
|
|
return true;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case RDCatchEvent::SetInputMonitorResponseOp:
|
|
if(f0.size()!=5) {
|
|
return false;
|
|
}
|
|
chan=f0.at(3).toInt(&ok);
|
|
if(ok&&(chan<255)) {
|
|
state=f0.at(4).toUInt(&ok);
|
|
if((state==0)||(state==1)) {
|
|
d_operation=op;
|
|
d_host_name=f0.at(1);
|
|
d_deck_channel=chan;
|
|
d_input_monitor_active=(state==1);
|
|
return true;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case RDCatchEvent::ReloadDecksOp:
|
|
if(f0.size()!=4) {
|
|
return false;
|
|
}
|
|
d_operation=op;
|
|
d_host_name=f0.at(1);
|
|
d_target_host_name=f0.at(3);
|
|
return true;
|
|
|
|
case RDCatchEvent::NullOp:
|
|
case RDCatchEvent::LastOp:
|
|
break;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
QString RDCatchEvent::write() const
|
|
{
|
|
QString ret;
|
|
|
|
//
|
|
// Common Fields
|
|
//
|
|
ret+="CATCH ";
|
|
ret+=d_host_name+" ";
|
|
ret+=QString::asprintf("%u",d_operation);
|
|
|
|
//
|
|
// Operation-specific Fields
|
|
//
|
|
switch(d_operation) {
|
|
case RDCatchEvent::DeckEventProcessedOp:
|
|
ret+=QString::asprintf(" %u",d_deck_channel);
|
|
ret+=QString::asprintf(" %u",d_event_number);
|
|
break;
|
|
|
|
case RDCatchEvent::DeckStatusResponseOp:
|
|
ret+=QString::asprintf(" %u",d_deck_channel);
|
|
ret+=QString::asprintf(" %u",d_deck_status);
|
|
ret+=QString::asprintf(" %u",d_event_id);
|
|
ret+=QString::asprintf(" %u",d_cart_number);
|
|
ret+=QString::asprintf(" %d",d_cut_number);
|
|
break;
|
|
|
|
case RDCatchEvent::DeckStatusQueryOp:
|
|
case RDCatchEvent::NullOp:
|
|
case RDCatchEvent::LastOp:
|
|
break;
|
|
|
|
case RDCatchEvent::SendMeterLevelsOp:
|
|
for(int i=0;i<d_meter_levels.size();i++) {
|
|
ret+=" "+d_meter_levels.at(i).dump();
|
|
}
|
|
break;
|
|
|
|
case RDCatchEvent::SetInputMonitorOp:
|
|
ret+=" "+d_target_host_name;
|
|
ret+=QString::asprintf(" %u",d_deck_channel);
|
|
ret+=QString::asprintf(" %u",d_input_monitor_active);
|
|
break;
|
|
|
|
case RDCatchEvent::SetInputMonitorResponseOp:
|
|
ret+=QString::asprintf(" %u",d_deck_channel);
|
|
ret+=QString::asprintf(" %u",d_input_monitor_active);
|
|
break;
|
|
|
|
case RDCatchEvent::StopDeckOp:
|
|
ret+=" "+d_target_host_name;
|
|
ret+=QString::asprintf(" %u",d_deck_channel);
|
|
break;
|
|
|
|
case RDCatchEvent::ReloadDecksOp:
|
|
ret+=" "+d_target_host_name;
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
QString RDCatchEvent::dump() const
|
|
{
|
|
QString ret;
|
|
|
|
//
|
|
// Common Fields
|
|
//
|
|
ret+="hostName: "+d_host_name+"\n";
|
|
|
|
//
|
|
// Operation-specific Fields
|
|
//
|
|
switch(d_operation) {
|
|
case RDCatchEvent::DeckEventProcessedOp:
|
|
ret+="operation: RDCatchEvent::DeckEventProcessedOp\n";
|
|
ret+=QString::asprintf("deck channel: %u\n",d_deck_channel);
|
|
ret+=QString::asprintf("event number: %u\n",d_event_number);
|
|
break;
|
|
|
|
case RDCatchEvent::SendMeterLevelsOp:
|
|
for(int i=0;i<d_meter_levels.size();i++) {
|
|
ret+="meter level: "+d_meter_levels.at(i).dump()+"\n";
|
|
}
|
|
break;
|
|
|
|
case RDCatchEvent::DeckStatusQueryOp:
|
|
ret+="operation: RDCatchEvent::DeckStatusQueryOp\n";
|
|
break;
|
|
|
|
case RDCatchEvent::DeckStatusResponseOp:
|
|
ret+="operation: RDCatchEvent::DeckStatusResponseOp\n";
|
|
ret+=QString::asprintf("deck channel: %u\n",d_deck_channel);
|
|
ret+=QString::asprintf("deck status: %u\n",d_deck_status);
|
|
ret+=QString::asprintf("event id: %u\n",d_event_id);
|
|
ret+=QString::asprintf("cart number: %u\n",d_cart_number);
|
|
ret+=QString::asprintf("cut number: %d\n",d_cut_number);
|
|
break;
|
|
|
|
case RDCatchEvent::ReloadDecksOp:
|
|
ret+="operation: RDCatchEvent::ReloadDecksOp\n";
|
|
ret+="target hostname: "+d_target_host_name+"\n";
|
|
break;
|
|
|
|
case RDCatchEvent::StopDeckOp:
|
|
ret+="operation: RDCatchEvent::StopDeckOp\n";
|
|
ret+="target hostname: "+d_target_host_name+"\n";
|
|
ret+=QString::asprintf("deck channel: %u\n",d_deck_channel);
|
|
break;
|
|
|
|
case RDCatchEvent::SetInputMonitorOp:
|
|
ret+="operation: RDCatchEvent::SetInputMonitorOp\n";
|
|
ret+="target hostname: "+d_target_host_name+"\n";
|
|
ret+=QString::asprintf("deck channel: %u\n",d_deck_channel);
|
|
ret+=QString::asprintf("input monitor active: %u\n",d_input_monitor_active);
|
|
break;
|
|
|
|
case RDCatchEvent::SetInputMonitorResponseOp:
|
|
ret+="operation: RDCatchEvent::SetInputMonitorResponseOp\n";
|
|
ret+=QString::asprintf("deck channel: %u\n",d_deck_channel);
|
|
ret+=QString::asprintf("input monitor active: %u\n",d_input_monitor_active);
|
|
break;
|
|
|
|
case RDCatchEvent::NullOp:
|
|
case RDCatchEvent::LastOp:
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
void RDCatchEvent::clear()
|
|
{
|
|
d_operation=RDCatchEvent::NullOp;
|
|
d_host_name=rda->station()->name();
|
|
d_target_host_name="";
|
|
d_event_id=0;
|
|
d_cart_number=0;
|
|
d_cut_number=0;
|
|
d_deck_channel=0;
|
|
d_event_number=0;
|
|
d_input_monitor_active=false;
|
|
d_deck_status=RDDeck::Offline;
|
|
d_meter_levels.clear();
|
|
}
|