| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | ///////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // Copyright (C) 2018 Edouard Griffiths, F4EXB                                   //
 | 
					
						
							|  |  |  | //                                                                               //
 | 
					
						
							|  |  |  | // This program is free software; you can redistribute it and/or modify          //
 | 
					
						
							|  |  |  | // it under the terms of the GNU General Public License as published by          //
 | 
					
						
							|  |  |  | // the Free Software Foundation as version 3 of the License, or                  //
 | 
					
						
							|  |  |  | //                                                                               //
 | 
					
						
							|  |  |  | // 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 V3 for more details.                               //
 | 
					
						
							|  |  |  | //                                                                               //
 | 
					
						
							|  |  |  | // You should have received a copy of the GNU General Public License             //
 | 
					
						
							|  |  |  | // along with this program. If not, see <http://www.gnu.org/licenses/>.          //
 | 
					
						
							|  |  |  | ///////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | #include <sys/time.h>
 | 
					
						
							|  |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2018-09-01 04:43:15 +02:00
										 |  |  | #include <boost/crc.hpp>
 | 
					
						
							|  |  |  | #include <boost/cstdint.hpp>
 | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-01 00:18:35 +02:00
										 |  |  | #include <QDebug>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | #include "SWGChannelSettings.h"
 | 
					
						
							|  |  |  | #include "SWGChannelReport.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-04 20:35:54 +02:00
										 |  |  | #include "SWGDaemonSourceReport.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-01 04:43:15 +02:00
										 |  |  | #include "dsp/devicesamplesink.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | #include "device/devicesinkapi.h"
 | 
					
						
							|  |  |  | #include "dsp/upchannelizer.h"
 | 
					
						
							|  |  |  | #include "dsp/threadedbasebandsamplesource.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:50:55 +02:00
										 |  |  | #include "daemonsourcethread.h"
 | 
					
						
							|  |  |  | #include "daemonsource.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | MESSAGE_CLASS_DEFINITION(DaemonSource::MsgSampleRateNotification, Message) | 
					
						
							|  |  |  | MESSAGE_CLASS_DEFINITION(DaemonSource::MsgConfigureDaemonSource, Message) | 
					
						
							|  |  |  | MESSAGE_CLASS_DEFINITION(DaemonSource::MsgQueryStreamData, Message) | 
					
						
							|  |  |  | MESSAGE_CLASS_DEFINITION(DaemonSource::MsgReportStreamData, Message) | 
					
						
							| 
									
										
										
										
											2018-09-01 00:18:35 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-12 01:21:51 +02:00
										 |  |  | const QString DaemonSource::m_channelIdURI = "sdrangel.channeltx.daemonsource"; | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | const QString DaemonSource::m_channelId ="DaemonSource"; | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | DaemonSource::DaemonSource(DeviceSinkAPI *deviceAPI) : | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  |     ChannelSourceAPI(m_channelIdURI), | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |     m_deviceAPI(deviceAPI), | 
					
						
							|  |  |  |     m_sourceThread(0), | 
					
						
							|  |  |  |     m_running(false), | 
					
						
							|  |  |  |     m_nbCorrectableErrors(0), | 
					
						
							|  |  |  |     m_nbUncorrectableErrors(0) | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     setObjectName(m_channelId); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 08:41:37 +02:00
										 |  |  |     connect(&m_dataQueue, SIGNAL(dataBlockEnqueued()), this, SLOT(handleData()), Qt::QueuedConnection); | 
					
						
							|  |  |  |     m_cm256p = m_cm256.isInitialized() ? &m_cm256 : 0; | 
					
						
							|  |  |  |     m_currentMeta.init(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  |     m_channelizer = new UpChannelizer(this); | 
					
						
							|  |  |  |     m_threadedChannelizer = new ThreadedBasebandSampleSource(m_channelizer, this); | 
					
						
							|  |  |  |     m_deviceAPI->addThreadedSource(m_threadedChannelizer); | 
					
						
							|  |  |  |     m_deviceAPI->addChannelAPI(this); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | DaemonSource::~DaemonSource() | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     m_deviceAPI->removeChannelAPI(this); | 
					
						
							|  |  |  |     m_deviceAPI->removeThreadedSource(m_threadedChannelizer); | 
					
						
							|  |  |  |     delete m_threadedChannelizer; | 
					
						
							|  |  |  |     delete m_channelizer; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | void DaemonSource::pull(Sample& sample) | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-09-13 02:33:56 +02:00
										 |  |  |     m_dataReadQueue.readSample(sample, true); // true is scale for Tx
 | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-13 00:45:03 +01:00
										 |  |  | void DaemonSource::pullAudio(int nbSamples) | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-11-13 00:45:03 +01:00
										 |  |  |     (void) nbSamples; | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | void DaemonSource::start() | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |     qDebug("DaemonSource::start"); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (m_running) { | 
					
						
							|  |  |  |         stop(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |     m_sourceThread = new DaemonSourceThread(&m_dataQueue); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |     m_sourceThread->startStop(true); | 
					
						
							|  |  |  |     m_sourceThread->dataBind(m_settings.m_dataAddress, m_settings.m_dataPort); | 
					
						
							|  |  |  |     m_running = true; | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | void DaemonSource::stop() | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |     qDebug("DaemonSource::stop"); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (m_sourceThread != 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_sourceThread->startStop(false); | 
					
						
							|  |  |  |         m_sourceThread->deleteLater(); | 
					
						
							|  |  |  |         m_sourceThread = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_running = false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | void DaemonSource::setDataLink(const QString& dataAddress, uint16_t dataPort) | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |     DaemonSourceSettings settings = m_settings; | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |     settings.m_dataAddress = dataAddress; | 
					
						
							|  |  |  |     settings.m_dataPort = dataPort; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |     MsgConfigureDaemonSource *msg = MsgConfigureDaemonSource::create(settings, false); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |     m_inputMessageQueue.push(msg); | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | bool DaemonSource::handleMessage(const Message& cmd) | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-09-01 00:18:35 +02:00
										 |  |  |     if (UpChannelizer::MsgChannelizerNotification::match(cmd)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         UpChannelizer::MsgChannelizerNotification& notif = (UpChannelizer::MsgChannelizerNotification&) cmd; | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |         qDebug() << "DaemonSource::handleMessage: MsgChannelizerNotification:" | 
					
						
							| 
									
										
										
										
											2018-09-01 00:18:35 +02:00
										 |  |  |                 << " basebandSampleRate: " << notif.getBasebandSampleRate() | 
					
						
							|  |  |  |                 << " outputSampleRate: " << notif.getSampleRate() | 
					
						
							|  |  |  |                 << " inputFrequencyOffset: " << notif.getFrequencyOffset(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (m_guiMessageQueue) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             MsgSampleRateNotification *msg = MsgSampleRateNotification::create(notif.getBasebandSampleRate()); | 
					
						
							|  |  |  |             m_guiMessageQueue->push(msg); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |     else if (MsgConfigureDaemonSource::match(cmd)) | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |         MsgConfigureDaemonSource& cfg = (MsgConfigureDaemonSource&) cmd; | 
					
						
							|  |  |  |         qDebug() << "MsgConfigureDaemonSource::handleMessage: MsgConfigureDaemonSource"; | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |         applySettings(cfg.getSettings(), cfg.getForce()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-02 19:12:03 +02:00
										 |  |  |     else if (MsgQueryStreamData::match(cmd)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (m_guiMessageQueue) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             struct timeval tv; | 
					
						
							|  |  |  |             gettimeofday(&tv, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             MsgReportStreamData *msg = MsgReportStreamData::create( | 
					
						
							|  |  |  |                     tv.tv_sec, | 
					
						
							|  |  |  |                     tv.tv_usec, | 
					
						
							|  |  |  |                     m_dataReadQueue.size(), | 
					
						
							|  |  |  |                     m_dataReadQueue.length(), | 
					
						
							|  |  |  |                     m_dataReadQueue.readSampleCount(), | 
					
						
							|  |  |  |                     m_nbCorrectableErrors, | 
					
						
							|  |  |  |                     m_nbUncorrectableErrors, | 
					
						
							|  |  |  |                     m_currentMeta.m_nbOriginalBlocks, | 
					
						
							|  |  |  |                     m_currentMeta.m_nbFECBlocks, | 
					
						
							|  |  |  |                     m_currentMeta.m_centerFrequency, | 
					
						
							|  |  |  |                     m_currentMeta.m_sampleRate); | 
					
						
							|  |  |  |             m_guiMessageQueue->push(msg); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | QByteArray DaemonSource::serialize() const | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     return m_settings.serialize(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-13 00:45:03 +01:00
										 |  |  | bool DaemonSource::deserialize(const QByteArray& data) | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-11-13 00:45:03 +01:00
										 |  |  |     (void) data; | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  |     if (m_settings.deserialize(data)) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |         MsgConfigureDaemonSource *msg = MsgConfigureDaemonSource::create(m_settings, true); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |         m_inputMessageQueue.push(msg); | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_settings.resetToDefaults(); | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |         MsgConfigureDaemonSource *msg = MsgConfigureDaemonSource::create(m_settings, true); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |         m_inputMessageQueue.push(msg); | 
					
						
							| 
									
										
										
										
											2018-08-31 23:29:53 +02:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | void DaemonSource::applySettings(const DaemonSourceSettings& settings, bool force) | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |     qDebug() << "DaemonSource::applySettings:" | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |             << " m_dataAddress: " << settings.m_dataAddress | 
					
						
							|  |  |  |             << " m_dataPort: " << settings.m_dataPort | 
					
						
							|  |  |  |             << " force: " << force; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool change = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ((m_settings.m_dataAddress != settings.m_dataAddress) || force) { | 
					
						
							|  |  |  |         change = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ((m_settings.m_dataPort != settings.m_dataPort) || force) { | 
					
						
							|  |  |  |         change = true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (change && m_sourceThread) { | 
					
						
							|  |  |  |         m_sourceThread->dataBind(settings.m_dataAddress, settings.m_dataPort); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_settings = settings; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-13 00:45:03 +01:00
										 |  |  | void DaemonSource::handleDataBlock(SDRDaemonDataBlock* dataBlock) | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-11-13 00:45:03 +01:00
										 |  |  |     (void) dataBlock; | 
					
						
							| 
									
										
										
										
											2018-09-01 04:43:15 +02:00
										 |  |  |     if (dataBlock->m_rxControlBlock.m_blockCount < SDRDaemonNbOrginalBlocks) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |         qWarning("DaemonSource::handleDataBlock: incomplete data block: not processing"); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:43:15 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         int blockCount = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (int blockIndex = 0; blockIndex < 256; blockIndex++) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if ((blockIndex == 0) && (dataBlock->m_rxControlBlock.m_metaRetrieved)) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 m_cm256DescriptorBlocks[blockCount].Index = 0; | 
					
						
							|  |  |  |                 m_cm256DescriptorBlocks[blockCount].Block = (void *) &(dataBlock->m_superBlocks[0].m_protectedBlock); | 
					
						
							|  |  |  |                 blockCount++; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else if (dataBlock->m_superBlocks[blockIndex].m_header.m_blockIndex != 0) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 m_cm256DescriptorBlocks[blockCount].Index = dataBlock->m_superBlocks[blockIndex].m_header.m_blockIndex; | 
					
						
							|  |  |  |                 m_cm256DescriptorBlocks[blockCount].Block = (void *) &(dataBlock->m_superBlocks[blockIndex].m_protectedBlock); | 
					
						
							|  |  |  |                 blockCount++; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |         //qDebug("DaemonSource::handleDataBlock: frame: %u blocks: %d", dataBlock.m_rxControlBlock.m_frameIndex, blockCount);
 | 
					
						
							| 
									
										
										
										
											2018-09-01 04:43:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Need to use the CM256 recovery
 | 
					
						
							|  |  |  |         if (m_cm256p &&(dataBlock->m_rxControlBlock.m_originalCount < SDRDaemonNbOrginalBlocks)) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |             qDebug("DaemonSource::handleDataBlock: %d recovery blocks", dataBlock->m_rxControlBlock.m_recoveryCount); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:43:15 +02:00
										 |  |  |             CM256::cm256_encoder_params paramsCM256; | 
					
						
							|  |  |  |             paramsCM256.BlockBytes = sizeof(SDRDaemonProtectedBlock); // never changes
 | 
					
						
							|  |  |  |             paramsCM256.OriginalCount = SDRDaemonNbOrginalBlocks;  // never changes
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (m_currentMeta.m_tv_sec == 0) { | 
					
						
							|  |  |  |                 paramsCM256.RecoveryCount = dataBlock->m_rxControlBlock.m_recoveryCount; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 paramsCM256.RecoveryCount = m_currentMeta.m_nbFECBlocks; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // update counters
 | 
					
						
							|  |  |  |             if (dataBlock->m_rxControlBlock.m_originalCount < SDRDaemonNbOrginalBlocks - paramsCM256.RecoveryCount) { | 
					
						
							|  |  |  |                 m_nbUncorrectableErrors += SDRDaemonNbOrginalBlocks - paramsCM256.RecoveryCount - dataBlock->m_rxControlBlock.m_originalCount; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 m_nbCorrectableErrors += dataBlock->m_rxControlBlock.m_recoveryCount; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (m_cm256.cm256_decode(paramsCM256, m_cm256DescriptorBlocks)) // CM256 decode
 | 
					
						
							|  |  |  |             { | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |                 qWarning() << "DaemonSource::handleDataBlock: decode CM256 error:" | 
					
						
							| 
									
										
										
										
											2018-09-01 04:43:15 +02:00
										 |  |  |                         << " m_originalCount: " << dataBlock->m_rxControlBlock.m_originalCount | 
					
						
							|  |  |  |                         << " m_recoveryCount: " << dataBlock->m_rxControlBlock.m_recoveryCount; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 for (int ir = 0; ir < dataBlock->m_rxControlBlock.m_recoveryCount; ir++) // restore missing blocks
 | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     int recoveryIndex = SDRDaemonNbOrginalBlocks - dataBlock->m_rxControlBlock.m_recoveryCount + ir; | 
					
						
							|  |  |  |                     int blockIndex = m_cm256DescriptorBlocks[recoveryIndex].Index; | 
					
						
							|  |  |  |                     SDRDaemonProtectedBlock *recoveredBlock = | 
					
						
							|  |  |  |                             (SDRDaemonProtectedBlock *) m_cm256DescriptorBlocks[recoveryIndex].Block; | 
					
						
							|  |  |  |                     memcpy((void *) &(dataBlock->m_superBlocks[blockIndex].m_protectedBlock), recoveredBlock, sizeof(SDRDaemonProtectedBlock)); | 
					
						
							|  |  |  |                     if ((blockIndex == 0) && !dataBlock->m_rxControlBlock.m_metaRetrieved) { | 
					
						
							|  |  |  |                         dataBlock->m_rxControlBlock.m_metaRetrieved = true; | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Validate block zero and retrieve its data
 | 
					
						
							|  |  |  |         if (dataBlock->m_rxControlBlock.m_metaRetrieved) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             SDRDaemonMetaDataFEC *metaData = (SDRDaemonMetaDataFEC *) &(dataBlock->m_superBlocks[0].m_protectedBlock); | 
					
						
							|  |  |  |             boost::crc_32_type crc32; | 
					
						
							|  |  |  |             crc32.process_bytes(metaData, 20); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (crc32.checksum() == metaData->m_crc32) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 if (!(m_currentMeta == *metaData)) | 
					
						
							|  |  |  |                 { | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |                     printMeta("DaemonSource::handleDataBlock", metaData); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:43:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     if (m_currentMeta.m_sampleRate != metaData->m_sampleRate) | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         m_channelizer->configure(m_channelizer->getInputMessageQueue(), metaData->m_sampleRate, 0); | 
					
						
							|  |  |  |                         m_dataReadQueue.setSize(calculateDataReadQueueSize(metaData->m_sampleRate)); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 m_currentMeta = *metaData; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |             { | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |                 qWarning() << "DaemonSource::handleDataBlock: recovered meta: invalid CRC32"; | 
					
						
							| 
									
										
										
										
											2018-09-01 04:43:15 +02:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         m_dataReadQueue.push(dataBlock); // Push into R/W buffer
 | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | void DaemonSource::handleData() | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     SDRDaemonDataBlock* dataBlock; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (m_running && ((dataBlock = m_dataQueue.pop()) != 0)) { | 
					
						
							|  |  |  |         handleDataBlock(dataBlock); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | void DaemonSource::printMeta(const QString& header, SDRDaemonMetaDataFEC *metaData) | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     qDebug().noquote() << header << ": " | 
					
						
							|  |  |  |             << "|" << metaData->m_centerFrequency | 
					
						
							|  |  |  |             << ":" << metaData->m_sampleRate | 
					
						
							|  |  |  |             << ":" << (int) (metaData->m_sampleBytes & 0xF) | 
					
						
							|  |  |  |             << ":" << (int) metaData->m_sampleBits | 
					
						
							|  |  |  |             << ":" << (int) metaData->m_nbOriginalBlocks | 
					
						
							|  |  |  |             << ":" << (int) metaData->m_nbFECBlocks | 
					
						
							|  |  |  |             << "|" << metaData->m_tv_sec | 
					
						
							|  |  |  |             << ":" << metaData->m_tv_usec | 
					
						
							|  |  |  |             << "|"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | uint32_t DaemonSource::calculateDataReadQueueSize(int sampleRate) | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     // scale for 20 blocks at 48 kS/s. Take next even number.
 | 
					
						
							|  |  |  |     uint32_t maxSize = sampleRate / 2400; | 
					
						
							|  |  |  |     maxSize = (maxSize % 2 == 0) ? maxSize : maxSize + 1; | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |     qDebug("DaemonSource::calculateDataReadQueueSize: set max queue size to %u blocks", maxSize); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |     return maxSize; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | int DaemonSource::webapiSettingsGet( | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |         SWGSDRangel::SWGChannelSettings& response, | 
					
						
							| 
									
										
										
										
											2018-11-13 00:45:03 +01:00
										 |  |  |         QString& errorMessage) | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-11-13 00:45:03 +01:00
										 |  |  |     (void) errorMessage; | 
					
						
							| 
									
										
										
										
											2018-09-04 20:35:54 +02:00
										 |  |  |     response.setDaemonSourceSettings(new SWGSDRangel::SWGDaemonSourceSettings()); | 
					
						
							|  |  |  |     response.getDaemonSourceSettings()->init(); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |     webapiFormatChannelSettings(response, m_settings); | 
					
						
							|  |  |  |     return 200; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | int DaemonSource::webapiSettingsPutPatch( | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |         bool force, | 
					
						
							|  |  |  |         const QStringList& channelSettingsKeys, | 
					
						
							|  |  |  |         SWGSDRangel::SWGChannelSettings& response, | 
					
						
							| 
									
										
										
										
											2018-11-13 00:45:03 +01:00
										 |  |  |         QString& errorMessage) | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-11-13 00:45:03 +01:00
										 |  |  |     (void) errorMessage; | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |     DaemonSourceSettings settings = m_settings; | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (channelSettingsKeys.contains("dataAddress")) { | 
					
						
							| 
									
										
										
										
											2018-09-04 20:35:54 +02:00
										 |  |  |         settings.m_dataAddress = *response.getDaemonSourceSettings()->getDataAddress(); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (channelSettingsKeys.contains("dataPort")) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2018-09-04 20:35:54 +02:00
										 |  |  |         int dataPort = response.getDaemonSourceSettings()->getDataPort(); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if ((dataPort < 1024) || (dataPort > 65535)) { | 
					
						
							|  |  |  |             settings.m_dataPort = 9090; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             settings.m_dataPort = dataPort; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-01 09:53:16 +02:00
										 |  |  |     if (channelSettingsKeys.contains("rgbColor")) { | 
					
						
							| 
									
										
										
										
											2018-09-04 20:35:54 +02:00
										 |  |  |         settings.m_rgbColor = response.getDaemonSourceSettings()->getRgbColor(); | 
					
						
							| 
									
										
										
										
											2018-09-01 09:53:16 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (channelSettingsKeys.contains("title")) { | 
					
						
							| 
									
										
										
										
											2018-09-04 20:35:54 +02:00
										 |  |  |         settings.m_title = *response.getDaemonSourceSettings()->getTitle(); | 
					
						
							| 
									
										
										
										
											2018-09-01 09:53:16 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |     MsgConfigureDaemonSource *msg = MsgConfigureDaemonSource::create(settings, force); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |     m_inputMessageQueue.push(msg); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |     qDebug("DaemonSource::webapiSettingsPutPatch: forward to GUI: %p", m_guiMessageQueue); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |     if (m_guiMessageQueue) // forward to GUI if any
 | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  |         MsgConfigureDaemonSource *msgToGUI = MsgConfigureDaemonSource::create(settings, force); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |         m_guiMessageQueue->push(msgToGUI); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     webapiFormatChannelSettings(response, settings); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 200; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | int DaemonSource::webapiReportGet( | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |         SWGSDRangel::SWGChannelReport& response, | 
					
						
							| 
									
										
										
										
											2018-11-13 00:45:03 +01:00
										 |  |  |         QString& errorMessage) | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-11-13 00:45:03 +01:00
										 |  |  |     (void) errorMessage; | 
					
						
							| 
									
										
										
										
											2018-09-04 20:35:54 +02:00
										 |  |  |     response.setDaemonSourceReport(new SWGSDRangel::SWGDaemonSourceReport()); | 
					
						
							|  |  |  |     response.getDaemonSourceReport()->init(); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |     webapiFormatChannelReport(response); | 
					
						
							|  |  |  |     return 200; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | void DaemonSource::webapiFormatChannelSettings(SWGSDRangel::SWGChannelSettings& response, const DaemonSourceSettings& settings) | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-09-04 20:35:54 +02:00
										 |  |  |     if (response.getDaemonSourceSettings()->getDataAddress()) { | 
					
						
							|  |  |  |         *response.getDaemonSourceSettings()->getDataAddress() = settings.m_dataAddress; | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2018-09-04 20:35:54 +02:00
										 |  |  |         response.getDaemonSourceSettings()->setDataAddress(new QString(settings.m_dataAddress)); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-04 20:35:54 +02:00
										 |  |  |     response.getDaemonSourceSettings()->setDataPort(settings.m_dataPort); | 
					
						
							|  |  |  |     response.getDaemonSourceSettings()->setRgbColor(settings.m_rgbColor); | 
					
						
							| 
									
										
										
										
											2018-09-01 09:53:16 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-04 20:35:54 +02:00
										 |  |  |     if (response.getDaemonSourceSettings()->getTitle()) { | 
					
						
							|  |  |  |         *response.getDaemonSourceSettings()->getTitle() = settings.m_title; | 
					
						
							| 
									
										
										
										
											2018-09-01 09:53:16 +02:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2018-09-04 20:35:54 +02:00
										 |  |  |         response.getDaemonSourceSettings()->setTitle(new QString(settings.m_title)); | 
					
						
							| 
									
										
										
										
											2018-09-01 09:53:16 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-11 16:32:14 +02:00
										 |  |  | void DaemonSource::webapiFormatChannelReport(SWGSDRangel::SWGChannelReport& response) | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     struct timeval tv; | 
					
						
							|  |  |  |     gettimeofday(&tv, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-04 20:35:54 +02:00
										 |  |  |     response.getDaemonSourceReport()->setTvSec(tv.tv_sec); | 
					
						
							|  |  |  |     response.getDaemonSourceReport()->setTvUSec(tv.tv_usec); | 
					
						
							|  |  |  |     response.getDaemonSourceReport()->setQueueSize(m_dataReadQueue.size()); | 
					
						
							|  |  |  |     response.getDaemonSourceReport()->setQueueLength(m_dataReadQueue.length()); | 
					
						
							|  |  |  |     response.getDaemonSourceReport()->setSamplesCount(m_dataReadQueue.readSampleCount()); | 
					
						
							|  |  |  |     response.getDaemonSourceReport()->setCorrectableErrorsCount(m_nbCorrectableErrors); | 
					
						
							|  |  |  |     response.getDaemonSourceReport()->setUncorrectableErrorsCount(m_nbUncorrectableErrors); | 
					
						
							|  |  |  |     response.getDaemonSourceReport()->setNbOriginalBlocks(m_currentMeta.m_nbOriginalBlocks); | 
					
						
							|  |  |  |     response.getDaemonSourceReport()->setNbFecBlocks(m_currentMeta.m_nbFECBlocks); | 
					
						
							|  |  |  |     response.getDaemonSourceReport()->setCenterFreq(m_currentMeta.m_centerFrequency); | 
					
						
							|  |  |  |     response.getDaemonSourceReport()->setSampleRate(m_currentMeta.m_sampleRate); | 
					
						
							| 
									
										
										
										
											2018-09-06 21:54:09 +02:00
										 |  |  |     response.getDaemonSourceReport()->setDeviceCenterFreq(m_deviceAPI->getSampleSink()->getCenterFrequency()/1000); | 
					
						
							|  |  |  |     response.getDaemonSourceReport()->setDeviceSampleRate(m_deviceAPI->getSampleSink()->getSampleRate()); | 
					
						
							| 
									
										
										
										
											2018-09-01 04:37:23 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-09-02 19:12:03 +02:00
										 |  |  | 
 |