mirror of
				https://github.com/ElvishArtisan/rivendell.git
				synced 2025-10-31 22:24:02 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			211 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			211 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // gvc7000.cpp
 | |
| //
 | |
| // A Rivendell switcher driver for Grass Valley Series 7000 Protocol
 | |
| //
 | |
| //   (C) Copyright 2002-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 <stdlib.h>
 | |
| #include <syslog.h>
 | |
| 
 | |
| #include <qsignalmapper.h>
 | |
| 
 | |
| #include <rdapplication.h>
 | |
| 
 | |
| #include "gvc7000.h"
 | |
| 
 | |
| Gvc7000::Gvc7000(RDMatrix *matrix,QObject *parent)
 | |
|   : Switcher(matrix,parent)
 | |
| {
 | |
|   gvc_matrix=matrix->matrix();
 | |
| 
 | |
|   //
 | |
|   // Get Matrix Parameters
 | |
|   //
 | |
|   gvc_ipaddress=matrix->ipAddress(RDMatrix::Primary);
 | |
|   gvc_ipport=matrix->ipPort(RDMatrix::Primary);
 | |
|   gvc_inputs=matrix->inputs();
 | |
|   gvc_outputs=matrix->outputs();
 | |
| 
 | |
|   //
 | |
|   // Keepalive Timer
 | |
|   //
 | |
|   gvc_keepalive_timer=new QTimer(this);
 | |
|   connect(gvc_keepalive_timer,SIGNAL(timeout()),this,SLOT(keepaliveData()));
 | |
|   gvc_keepalive_timer->start(GVC7000_KEEPALIVE_INTERVAL);
 | |
| 
 | |
|   //
 | |
|   // Reconnection Timer
 | |
|   //
 | |
|   gvc_reconnect_timer=new QTimer(this);
 | |
|   gvc_reconnect_timer->setSingleShot(true);
 | |
|   connect(gvc_reconnect_timer,SIGNAL(timeout()),this,SLOT(ipConnect()));
 | |
| 
 | |
|   //
 | |
|   // Initialize the connection
 | |
|   //
 | |
|   gvc_socket=new QTcpSocket(this);
 | |
|   connect(gvc_socket,SIGNAL(connected()),this,SLOT(connectedData()));
 | |
|   connect(gvc_socket,SIGNAL(disconnected()),this,SLOT(disconnectedData()));
 | |
|   connect(gvc_socket,SIGNAL(error(QAbstractSocket::SocketError)),
 | |
| 	  this,SLOT(errorData(QAbstractSocket::SocketError)));
 | |
|   ipConnect();
 | |
| }
 | |
| 
 | |
| 
 | |
| RDMatrix::Type Gvc7000::type()
 | |
| {
 | |
|   return RDMatrix::Gvc7000;
 | |
| }
 | |
| 
 | |
| 
 | |
| unsigned Gvc7000::gpiQuantity()
 | |
| {
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| unsigned Gvc7000::gpoQuantity()
 | |
| {
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| bool Gvc7000::primaryTtyActive()
 | |
| {
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| 
 | |
| bool Gvc7000::secondaryTtyActive()
 | |
| {
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| 
 | |
| void Gvc7000::processCommand(RDMacro *cmd)
 | |
| {
 | |
|   switch(cmd->command()) {
 | |
|   case RDMacro::ST:
 | |
|     if((cmd->arg(1).toInt()<1)||(cmd->arg(1).toInt()>gvc_inputs)||
 | |
|        (cmd->arg(2).toInt()<1)||(cmd->arg(2).toInt()>gvc_outputs)) {
 | |
|       cmd->acknowledge(false);
 | |
|       emit rmlEcho(cmd);
 | |
|       return;
 | |
|     }
 | |
|     SendCommand(ToSeries7000Native(QString().sprintf("TI,%04X,%04X",cmd->arg(2).toInt()-1,cmd->arg(1).toInt()-1)));
 | |
|     cmd->acknowledge(true);
 | |
|     emit rmlEcho(cmd);
 | |
|     break;
 | |
| 
 | |
|   default:
 | |
|     cmd->acknowledge(false);
 | |
|     emit rmlEcho(cmd);
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| void Gvc7000::ipConnect()
 | |
| {
 | |
|   gvc_socket->connectToHost(gvc_ipaddress.toString(),gvc_ipport);
 | |
| }
 | |
| 
 | |
| 
 | |
| void Gvc7000::keepaliveData()
 | |
| {
 | |
|   SendCommand(ToSeries7000Native("QJ"));
 | |
| }
 | |
| 
 | |
| 
 | |
| void Gvc7000::connectedData()
 | |
| {
 | |
|   rda->syslog(LOG_INFO,"connection to Gvc7000 device at %s:%d established",
 | |
| 	      (const char *)gvc_ipaddress.toString().toUtf8(),gvc_ipport);
 | |
| }
 | |
| 
 | |
| 
 | |
| void Gvc7000::disconnectedData()
 | |
| {
 | |
|   rda->syslog(LOG_WARNING,
 | |
| 	 "connection to Gvc7000 device at %s:%d closed unexpectedly, attempting reconnect",
 | |
| 	      (const char *)gvc_ipaddress.toString().toUtf8(),gvc_ipport);
 | |
|   gvc_reconnect_timer->start(GVC7000_RECONNECT_INTERVAL,true);
 | |
| }
 | |
| 
 | |
| 
 | |
| void Gvc7000::errorData(QAbstractSocket::SocketError err)
 | |
| {
 | |
|   switch(err) {
 | |
|   case QAbstractSocket::ConnectionRefusedError:
 | |
|     rda->syslog(LOG_WARNING,
 | |
| 	  "connection to Gvc7000 device at %s:%d refused, attempting reconnect",
 | |
| 		(const char *)gvc_ipaddress.toString().toUtf8(),gvc_ipport);
 | |
|     gvc_reconnect_timer->start(GVC7000_RECONNECT_INTERVAL,true);
 | |
|     break;
 | |
| 
 | |
|   case QAbstractSocket::HostNotFoundError:
 | |
|     rda->syslog(LOG_WARNING,
 | |
| 	       "error on connection to Gvc7000 device at %s:%d: Host Not Found",
 | |
| 		(const char *)gvc_ipaddress.toString().toUtf8(),
 | |
| 		gvc_ipport);
 | |
|     break;
 | |
| 
 | |
|   case QAbstractSocket::RemoteHostClosedError:
 | |
|     break;
 | |
| 
 | |
|   default:
 | |
|     rda->syslog(LOG_WARNING,
 | |
| 	    "socket error %d on connection to Gvc7000 device at %s:%d: Socket Read Error",
 | |
| 		err,
 | |
| 		(const char *)gvc_ipaddress.toString().toUtf8(),gvc_ipport);
 | |
|     break;
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| void Gvc7000::SendCommand(const QString &str)
 | |
| {
 | |
|   syslog(LOG_DEBUG,"gvc7000 sending \"%s\"",(const char *)str.toAscii());
 | |
|   gvc_socket->write(str.toAscii());
 | |
| }
 | |
| 
 | |
| 
 | |
| QString Gvc7000::ToSeries7000Native(const QString &str) const
 | |
| {
 | |
|   QString ret=str;
 | |
| 
 | |
|   ret.replace(",","\t");
 | |
|   if(ret.right(1)!="\t") {
 | |
|     ret+="\t";
 | |
|   }
 | |
| 
 | |
|   QString msg=QString("N")+  // Protocol ID
 | |
|     "0"+                     // Sequence Flag
 | |
|     ret;                     // Data
 | |
| 
 | |
|   uint8_t sum=0;
 | |
|   for(int i=0;i<msg.length();i++) {
 | |
|     sum+=msg.at(i).toAscii();
 | |
|   }
 | |
|   sum=0x100-sum;
 | |
| 
 | |
|   return QChar(1)+                  // Header
 | |
|     msg+                            // Message
 | |
|     QString().sprintf("%02X",sum)+  // Checksum
 | |
|     QChar(4);                       // Footer
 | |
| }
 |