From 923ac7695a7ec7e5f1025889d2b8adf40b9fcbcc Mon Sep 17 00:00:00 2001 From: f4exb Date: Wed, 11 Dec 2019 23:19:22 +0100 Subject: [PATCH] Remote Sink: threading refactoring --- plugins/channelrx/CMakeLists.txt | 6 +- plugins/channelrx/remotesink/CMakeLists.txt | 6 +- plugins/channelrx/remotesink/remotesink.cpp | 7 +- .../remotesink/remotesinkbaseband.cpp | 6 +- .../channelrx/remotesink/remotesinkbaseband.h | 6 +- .../channelrx/remotesink/remotesinkfifo.cpp | 95 +++++++++++++ plugins/channelrx/remotesink/remotesinkfifo.h | 55 ++++++++ .../channelrx/remotesink/remotesinkgui.cpp | 4 +- ...otesinkthread.cpp => remotesinksender.cpp} | 129 ++++++------------ ...{remotesinkthread.h => remotesinksender.h} | 55 ++------ .../channelrx/remotesink/remotesinksink.cpp | 115 +++++++--------- plugins/channelrx/remotesink/remotesinksink.h | 32 ++--- 12 files changed, 293 insertions(+), 223 deletions(-) create mode 100644 plugins/channelrx/remotesink/remotesinkfifo.cpp create mode 100644 plugins/channelrx/remotesink/remotesinkfifo.h rename plugins/channelrx/remotesink/{remotesinkthread.cpp => remotesinksender.cpp} (62%) rename plugins/channelrx/remotesink/{remotesinkthread.h => remotesinksender.h} (66%) diff --git a/plugins/channelrx/CMakeLists.txt b/plugins/channelrx/CMakeLists.txt index 5c17b91e8..60f72906d 100644 --- a/plugins/channelrx/CMakeLists.txt +++ b/plugins/channelrx/CMakeLists.txt @@ -13,9 +13,9 @@ if(LIBDSDCC_FOUND AND LIBMBE_FOUND) add_subdirectory(demoddsd) endif(LIBDSDCC_FOUND AND LIBMBE_FOUND) -# if(CM256CC_FOUND) -# add_subdirectory(remotesink) -# endif(CM256CC_FOUND) +if(CM256CC_FOUND) + add_subdirectory(remotesink) +endif(CM256CC_FOUND) if (CODEC2_FOUND) add_subdirectory(demodfreedv) diff --git a/plugins/channelrx/remotesink/CMakeLists.txt b/plugins/channelrx/remotesink/CMakeLists.txt index b6845bab5..172855119 100644 --- a/plugins/channelrx/remotesink/CMakeLists.txt +++ b/plugins/channelrx/remotesink/CMakeLists.txt @@ -15,7 +15,8 @@ set(remotesink_SOURCES remotesinksink.cpp remotesinksettings.cpp remotesinkwebapiadapter.cpp - remotesinkthread.cpp + remotesinksender.cpp + remotesinkfifo.cpp remotesinkplugin.cpp ) @@ -25,7 +26,8 @@ set(remotesink_HEADERS remotesinksink.h remotesinksettings.h remotesinkwebapiadapter.h - remotesinkthread.h + remotesinksender.h + remotesinkfifo.h remotesinkplugin.h ) diff --git a/plugins/channelrx/remotesink/remotesink.cpp b/plugins/channelrx/remotesink/remotesink.cpp index 5ac9e2b3f..30b35860f 100644 --- a/plugins/channelrx/remotesink/remotesink.cpp +++ b/plugins/channelrx/remotesink/remotesink.cpp @@ -34,6 +34,7 @@ #include "dsp/dspcommands.h" #include "dsp/hbfilterchainconverter.h" #include "dsp/devicesamplemimo.h" +#include "dsp/dspdevicesourceengine.h" #include "device/deviceapi.h" #include "remotesinkbaseband.h" @@ -95,13 +96,13 @@ void RemoteSink::start() } m_thread->start(); - m_basebandSink->startSink(); + m_basebandSink->startSender(); } void RemoteSink::stop() { qDebug("RemoteSink::stop"); - m_basebandSink->stopSink(); + m_basebandSink->stopSender(); m_thread->exit(); m_thread->wait(); } @@ -130,7 +131,7 @@ bool RemoteSink::handleMessage(const Message& cmd) if (getMessageQueueToGUI()) { DSPSignalNotification* msgToGUI = new DSPSignalNotification(notif); // make a copy - getMessageQueueToGUI()->push(msgToBaseband); + getMessageQueueToGUI()->push(msgToGUI); } return true; diff --git a/plugins/channelrx/remotesink/remotesinkbaseband.cpp b/plugins/channelrx/remotesink/remotesinkbaseband.cpp index 2016c8ac5..3467cef8e 100644 --- a/plugins/channelrx/remotesink/remotesinkbaseband.cpp +++ b/plugins/channelrx/remotesink/remotesinkbaseband.cpp @@ -116,7 +116,8 @@ bool RemoteSinkBaseband::handleMessage(const Message& cmd) m_basebandSampleRate = notif.getSampleRate(); qDebug() << "RemoteSinkBaseband::handleMessage: DSPSignalNotification: basebandSampleRate:" << m_basebandSampleRate; m_channelizer->setBasebandSampleRate(m_basebandSampleRate); - m_sink.applySampleRate(m_basebandSampleRate/ (1<setDecimation(settings.m_log2Decim, settings.m_filterChainHash); - m_sink.applySampleRate(m_basebandSampleRate/ (1<setBasebandSampleRate(m_basebandSampleRate); - m_sink.applySampleRate(m_basebandSampleRate/ (1<. // +/////////////////////////////////////////////////////////////////////////////////// + +#include "remotesinkfifo.h" + +RemoteSinkFifo::RemoteSinkFifo(QObject *parent) : + QObject(parent) +{} + +RemoteSinkFifo::RemoteSinkFifo(unsigned int size, QObject *parent) : + QObject(parent) +{ + resize(size); +} + +RemoteSinkFifo::~RemoteSinkFifo() +{} + +void RemoteSinkFifo::resize(unsigned int size) +{ + QMutexLocker mutexLocker(&m_mutex); + m_size = size; + m_data.resize(m_size); + m_readHead = 0; + m_servedHead = 0; + m_writeHead = 0; +} + +void RemoteSinkFifo::reset() +{ + m_readHead = 0; + m_servedHead = 0; + m_writeHead = 0; +} + +RemoteDataBlock *RemoteSinkFifo::getDataBlock() +{ + QMutexLocker mutexLocker(&m_mutex); + m_servedHead = m_writeHead; + + if (m_writeHead < m_size - 1) { + m_writeHead++; + } else { + m_writeHead = 0; + } + + emit dataBlockServed(); + return &m_data[m_servedHead]; +} + +unsigned int RemoteSinkFifo::readDataBlock(RemoteDataBlock **dataBlock) +{ + QMutexLocker mutexLocker(&m_mutex); + + if (calculateRemainder() == 0) + { + *dataBlock = nullptr; + return 0; + } + else + { + *dataBlock = &m_data[m_readHead]; + m_readHead = m_readHead < m_size - 1 ? m_readHead + 1 : 0; + return calculateRemainder(); + } +} + +unsigned int RemoteSinkFifo::getRemainder() +{ + QMutexLocker mutexLocker(&m_mutex); + return calculateRemainder(); +} + +unsigned int RemoteSinkFifo::calculateRemainder() +{ + if (m_readHead <= m_servedHead) { + return m_servedHead - m_readHead; + } else { + return m_size - (m_readHead - m_servedHead); + } +} \ No newline at end of file diff --git a/plugins/channelrx/remotesink/remotesinkfifo.h b/plugins/channelrx/remotesink/remotesinkfifo.h new file mode 100644 index 000000000..e1e055320 --- /dev/null +++ b/plugins/channelrx/remotesink/remotesinkfifo.h @@ -0,0 +1,55 @@ +/////////////////////////////////////////////////////////////////////////////////// +// Copyright (C) 2019 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 // +// (at your option) any later version. // +// // +// 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 . // +/////////////////////////////////////////////////////////////////////////////////// + +#ifndef REMOTESINK_REMOTESINKFIFO_H_ +#define REMOTESINK_REMOTESINKFIFO_H_ + +#include + +#include +#include + +#include "channel/remotedatablock.h" + +class RemoteSinkFifo : public QObject { + Q_OBJECT +public: + RemoteSinkFifo(QObject *parent = nullptr); + RemoteSinkFifo(unsigned int size, QObject *parent = nullptr); + ~RemoteSinkFifo(); + void resize(unsigned int size); + void reset(); + + RemoteDataBlock *getDataBlock(); + unsigned int readDataBlock(RemoteDataBlock **dataBlock); + unsigned int getRemainder(); + +signals: + void dataBlockServed(); + +private: + std::vector m_data; + int m_size; + unsigned int m_readHead; //!< index of last data block processed + unsigned int m_servedHead; //!< index of last data block served + unsigned int m_writeHead; //!< index of next data block to serve + QMutex m_mutex; + + unsigned int calculateRemainder(); +}; + +#endif // REMOTESINK_REMOTESINKFIFO_H_ \ No newline at end of file diff --git a/plugins/channelrx/remotesink/remotesinkgui.cpp b/plugins/channelrx/remotesink/remotesinkgui.cpp index 8e1725875..123e75b00 100644 --- a/plugins/channelrx/remotesink/remotesinkgui.cpp +++ b/plugins/channelrx/remotesink/remotesinkgui.cpp @@ -296,6 +296,7 @@ void RemoteSinkGUI::onMenuDialogCalled(const QPoint &p) void RemoteSinkGUI::on_decimationFactor_currentIndexChanged(int index) { m_settings.m_log2Decim = index; + updateTxDelayTime(); applyDecimation(); } @@ -368,7 +369,8 @@ void RemoteSinkGUI::updateTxDelayTime() { double txDelayRatio = m_settings.m_txDelay / 100.0; int samplesPerBlock = RemoteNbBytesPerBlock / sizeof(Sample); - double delay = m_basebandSampleRate == 0 ? 0.0 : (127*samplesPerBlock*txDelayRatio) / m_basebandSampleRate; + int channelSampleRate = m_basebandSampleRate / (1<txDelayTime->setText(tr("%1µs").arg(QString::number(delay*1e6, 'f', 0))); } diff --git a/plugins/channelrx/remotesink/remotesinkthread.cpp b/plugins/channelrx/remotesink/remotesinksender.cpp similarity index 62% rename from plugins/channelrx/remotesink/remotesinkthread.cpp rename to plugins/channelrx/remotesink/remotesinksender.cpp index 28f44c662..b7b1e74e0 100644 --- a/plugins/channelrx/remotesink/remotesinkthread.cpp +++ b/plugins/channelrx/remotesink/remotesinksender.cpp @@ -21,90 +21,73 @@ // along with this program. If not, see . // /////////////////////////////////////////////////////////////////////////////////// -#include "remotesinkthread.h" -#include +#include +#include + #include #include "cm256cc/cm256.h" -MESSAGE_CLASS_DEFINITION(RemoteSinkThread::MsgStartStop, Message) +#include "channel/remotedatablock.h" +#include "remotesinksender.h" -RemoteSinkThread::RemoteSinkThread(QObject* parent) : - QThread(parent), - m_running(false), +RemoteSinkSender::RemoteSinkSender() : + m_fifo(20, this), m_address(QHostAddress::LocalHost), - m_socket(0) + m_socket(nullptr) { + qDebug("RemoteSinkSender::RemoteSinkSender"); + m_cm256p = m_cm256.isInitialized() ? &m_cm256 : nullptr; + m_socket = new QUdpSocket(this); - m_cm256p = m_cm256.isInitialized() ? &m_cm256 : 0; - connect(&m_inputMessageQueue, SIGNAL(messageEnqueued()), this, SLOT(handleInputMessages()), Qt::QueuedConnection); + QObject::connect( + &m_fifo, + &RemoteSinkFifo::dataBlockServed, + this, + &RemoteSinkSender::handleData, + Qt::QueuedConnection + ); } -RemoteSinkThread::~RemoteSinkThread() +RemoteSinkSender::~RemoteSinkSender() { - qDebug("RemoteSinkThread::~RemoteSinkThread"); -} - -void RemoteSinkThread::startStop(bool start) -{ - MsgStartStop *msg = MsgStartStop::create(start); - m_inputMessageQueue.push(msg); -} - -void RemoteSinkThread::startWork() -{ - qDebug("RemoteSinkThread::startWork"); - m_startWaitMutex.lock(); - m_socket = new QUdpSocket(this); - start(); - while(!m_running) - m_startWaiter.wait(&m_startWaitMutex, 100); - m_startWaitMutex.unlock(); -} - -void RemoteSinkThread::stopWork() -{ - qDebug("RemoteSinkThread::stopWork"); + qDebug("RemoteSinkSender::~RemoteSinkSender"); delete m_socket; - m_socket = 0; - m_running = false; - wait(); } -void RemoteSinkThread::run() +RemoteDataBlock *RemoteSinkSender::getDataBlock() { - qDebug("RemoteSinkThread::run: begin"); - m_running = true; - m_startWaiter.wakeAll(); + return m_fifo.getDataBlock(); +} - while (m_running) +void RemoteSinkSender::handleData() +{ + RemoteDataBlock *dataBlock; + unsigned int remainder = m_fifo.getRemainder(); + + while (remainder != 0) { - sleep(1); // Do nothing as everything is in the data handler (dequeuer) + remainder = m_fifo.readDataBlock(&dataBlock); + + if (dataBlock) { + sendDataBlock(dataBlock); + } } - - m_running = false; - qDebug("RemoteSinkThread::run: end"); } -void RemoteSinkThread::processDataBlock(RemoteDataBlock *dataBlock) -{ - handleDataBlock(*dataBlock); - delete dataBlock; -} - -void RemoteSinkThread::handleDataBlock(RemoteDataBlock& dataBlock) +void RemoteSinkSender::sendDataBlock(RemoteDataBlock *dataBlock) { CM256::cm256_encoder_params cm256Params; //!< Main interface with CM256 encoder CM256::cm256_block descriptorBlocks[256]; //!< Pointers to data for CM256 encoder RemoteProtectedBlock fecBlocks[256]; //!< FEC data - uint16_t frameIndex = dataBlock.m_txControlBlock.m_frameIndex; - int nbBlocksFEC = dataBlock.m_txControlBlock.m_nbBlocksFEC; - int txDelay = dataBlock.m_txControlBlock.m_txDelay; - m_address.setAddress(dataBlock.m_txControlBlock.m_dataAddress); - uint16_t dataPort = dataBlock.m_txControlBlock.m_dataPort; - RemoteSuperBlock *txBlockx = dataBlock.m_superBlocks; + uint16_t frameIndex = dataBlock->m_txControlBlock.m_frameIndex; + int nbBlocksFEC = dataBlock->m_txControlBlock.m_nbBlocksFEC; + int txDelay = dataBlock->m_txControlBlock.m_txDelay; + m_address.setAddress(dataBlock->m_txControlBlock.m_dataAddress); + uint16_t dataPort = dataBlock->m_txControlBlock.m_dataPort; + RemoteSuperBlock *txBlockx = dataBlock->m_superBlocks; if ((nbBlocksFEC == 0) || !m_cm256p) // Do not FEC encode { @@ -114,7 +97,7 @@ void RemoteSinkThread::handleDataBlock(RemoteDataBlock& dataBlock) { // send block via UDP m_socket->writeDatagram((const char*)&txBlockx[i], (qint64 ) RemoteUdpSize, m_address, dataPort); - //usleep(txDelay); + std::this_thread::sleep_for(std::chrono::microseconds(txDelay)); } } } @@ -142,7 +125,7 @@ void RemoteSinkThread::handleDataBlock(RemoteDataBlock& dataBlock) // Encode FEC blocks if (m_cm256p->cm256_encode(cm256Params, descriptorBlocks, fecBlocks)) { - qWarning("RemoteSinkThread::handleDataBlock: CM256 encode failed. No transmission."); + qWarning("RemoteSinkSender::handleDataBlock: CM256 encode failed. No transmission."); // TODO: send without FEC changing meta data to set indication of no FEC } @@ -159,32 +142,10 @@ void RemoteSinkThread::handleDataBlock(RemoteDataBlock& dataBlock) { // send block via UDP m_socket->writeDatagram((const char*)&txBlockx[i], (qint64 ) RemoteUdpSize, m_address, dataPort); - //usleep(txDelay); + std::this_thread::sleep_for(std::chrono::microseconds(txDelay)); } } } - dataBlock.m_txControlBlock.m_processed = true; -} - -void RemoteSinkThread::handleInputMessages() -{ - Message* message; - - while ((message = m_inputMessageQueue.pop()) != 0) - { - if (MsgStartStop::match(*message)) - { - MsgStartStop* notif = (MsgStartStop*) message; - qDebug("RemoteSinkThread::handleInputMessages: MsgStartStop: %s", notif->getStartStop() ? "start" : "stop"); - - if (notif->getStartStop()) { - startWork(); - } else { - stopWork(); - } - - delete message; - } - } + dataBlock->m_txControlBlock.m_processed = true; } diff --git a/plugins/channelrx/remotesink/remotesinkthread.h b/plugins/channelrx/remotesink/remotesinksender.h similarity index 66% rename from plugins/channelrx/remotesink/remotesinkthread.h rename to plugins/channelrx/remotesink/remotesinksender.h index cfe3af953..98d74279c 100644 --- a/plugins/channelrx/remotesink/remotesinkthread.h +++ b/plugins/channelrx/remotesink/remotesinksender.h @@ -21,10 +21,10 @@ // along with this program. If not, see . // /////////////////////////////////////////////////////////////////////////////////// -#ifndef PLUGINS_CHANNELRX_REMOTESINK_REMOTESINKTHREAD_H_ -#define PLUGINS_CHANNELRX_REMOTESINK_REMOTESINKTHREAD_H_ +#ifndef PLUGINS_CHANNELRX_REMOTESINK_REMOTESINKSENDER_H_ +#define PLUGINS_CHANNELRX_REMOTESINK_REMOTESINKSENDER_H_ -#include +#include #include #include #include @@ -34,63 +34,34 @@ #include "util/message.h" #include "util/messagequeue.h" +#include "remotesinkfifo.h" + class RemoteDataBlock; class CM256; class QUdpSocket; -class RemoteSinkThread : public QThread { +class RemoteSinkSender : public QObject { Q_OBJECT public: - class MsgStartStop : public Message { - MESSAGE_CLASS_DECLARATION + RemoteSinkSender(); + ~RemoteSinkSender(); - public: - bool getStartStop() const { return m_startStop; } - - static MsgStartStop* create(bool startStop) { - return new MsgStartStop(startStop); - } - - protected: - bool m_startStop; - - MsgStartStop(bool startStop) : - Message(), - m_startStop(startStop) - { } - }; - - RemoteSinkThread(QObject* parent = 0); - ~RemoteSinkThread(); - - void startStop(bool start); - -public slots: - void processDataBlock(RemoteDataBlock *dataBlock); + RemoteDataBlock *getDataBlock(); private: - QMutex m_startWaitMutex; - QWaitCondition m_startWaiter; - volatile bool m_running; - + RemoteSinkFifo m_fifo; CM256 m_cm256; CM256 *m_cm256p; QHostAddress m_address; QUdpSocket *m_socket; - MessageQueue m_inputMessageQueue; - - void startWork(); - void stopWork(); - - void run(); - void handleDataBlock(RemoteDataBlock& dataBlock); + void sendDataBlock(RemoteDataBlock *dataBlock); private slots: - void handleInputMessages(); + void handleData(); }; -#endif // PLUGINS_CHANNELRX_REMOTESINK_REMOTESINKTHREAD_H_ +#endif // PLUGINS_CHANNELRX_REMOTESINK_REMOTESINKSENDER_H_ diff --git a/plugins/channelrx/remotesink/remotesinksink.cpp b/plugins/channelrx/remotesink/remotesinksink.cpp index 94e76faeb..883856299 100644 --- a/plugins/channelrx/remotesink/remotesinksink.cpp +++ b/plugins/channelrx/remotesink/remotesinksink.cpp @@ -16,6 +16,7 @@ /////////////////////////////////////////////////////////////////////////////////// #include +#include #include #include @@ -23,47 +24,64 @@ #include "dsp/hbfilterchainconverter.h" #include "util/timeutil.h" -#include "remotesinkthread.h" +#include "remotesinksender.h" #include "remotesinksink.h" RemoteSinkSink::RemoteSinkSink() : - m_running(false), - m_remoteSinkThread(nullptr), m_txBlockIndex(0), m_frameCount(0), m_sampleIndex(0), m_dataBlock(nullptr), - m_centerFrequency(0), + m_deviceCenterFrequency(0), m_frequencyOffset(0), - m_sampleRate(48000), + m_basebandSampleRate(48000), m_nbBlocksFEC(0), m_txDelay(35), m_dataAddress("127.0.0.1"), m_dataPort(9090) { + qDebug("RemoteSinkSink::RemoteSinkSink"); + + m_senderThread = new QThread(this); + m_remoteSinkSender = new RemoteSinkSender(); + m_remoteSinkSender->moveToThread(m_senderThread); + applySettings(m_settings, true); } RemoteSinkSink::~RemoteSinkSink() { - QMutexLocker mutexLocker(&m_dataBlockMutex); + qDebug("RemoteSinkSink::~RemoteSinkSink"); - if (m_dataBlock && !m_dataBlock->m_txControlBlock.m_complete) { - delete m_dataBlock; - } + delete m_remoteSinkSender; + delete m_senderThread; } -void RemoteSinkSink::setTxDelay(int txDelay, int nbBlocksFEC) +void RemoteSinkSink::startSender() +{ + qDebug("RemoteSinkSink::startSender"); + m_senderThread->start(); +} + +void RemoteSinkSink::stopSender() +{ + qDebug("RemoteSinkSink::stopSender"); + m_senderThread->exit(); + m_senderThread->wait(); +} + +void RemoteSinkSink::setTxDelay(int txDelay, int nbBlocksFEC, int log2Decim) { double txDelayRatio = txDelay / 100.0; int samplesPerBlock = RemoteNbBytesPerBlock / sizeof(Sample); - double delay = m_sampleRate == 0 ? 1.0 : (127*samplesPerBlock*txDelayRatio) / m_sampleRate; + int sampleRate = m_basebandSampleRate / (1<getDataBlock(); // ask a new block to sender } boost::crc_32_type crc32; @@ -158,7 +176,6 @@ void RemoteSinkSink::feed(const SampleVector::const_iterator& begin, const Sampl if (m_txBlockIndex == RemoteNbOrginalBlocks - 1) // frame complete { - m_dataBlockMutex.lock(); m_dataBlock->m_txControlBlock.m_frameIndex = m_frameCount; m_dataBlock->m_txControlBlock.m_processed = false; m_dataBlock->m_txControlBlock.m_complete = true; @@ -167,9 +184,7 @@ void RemoteSinkSink::feed(const SampleVector::const_iterator& begin, const Sampl m_dataBlock->m_txControlBlock.m_dataAddress = m_dataAddress; m_dataBlock->m_txControlBlock.m_dataPort = m_dataPort; - emit dataBlockAvailable(m_dataBlock); - m_dataBlock = new RemoteDataBlock(); // create a new one immediately - m_dataBlockMutex.unlock(); + m_dataBlock = m_remoteSinkSender->getDataBlock(); // ask a new block to sender m_txBlockIndex = 0; m_frameCount++; @@ -182,39 +197,6 @@ void RemoteSinkSink::feed(const SampleVector::const_iterator& begin, const Sampl } } -void RemoteSinkSink::start() -{ - qDebug("RemoteSinkSink::start"); - - memset((void *) &m_currentMetaFEC, 0, sizeof(RemoteMetaDataFEC)); - - if (m_running) { - stop(); - } - - m_remoteSinkThread = new RemoteSinkThread(); - connect(this, - SIGNAL(dataBlockAvailable(RemoteDataBlock *)), - m_remoteSinkThread, - SLOT(processDataBlock(RemoteDataBlock *)), - Qt::QueuedConnection); - m_remoteSinkThread->startStop(true); - m_running = true; -} - -void RemoteSinkSink::stop() -{ - qDebug("RemoteSinkSink::stop"); - - if (m_remoteSinkThread) - { - m_remoteSinkThread->startStop(false); - m_remoteSinkThread->deleteLater(); - } - - m_running = false; -} - void RemoteSinkSink::applySettings(const RemoteSinkSettings& settings, bool force) { qDebug() << "RemoteSinkSink::applySettings:" @@ -225,16 +207,6 @@ void RemoteSinkSink::applySettings(const RemoteSinkSettings& settings, bool forc << " m_streamIndex: " << settings.m_streamIndex << " force: " << force; - if ((m_settings.m_nbFECBlocks != settings.m_nbFECBlocks) || force) - { - setNbBlocksFEC(settings.m_nbFECBlocks); - setTxDelay(settings.m_txDelay, settings.m_nbFECBlocks); - } - - if ((m_settings.m_txDelay != settings.m_txDelay) || force) { - setTxDelay(settings.m_txDelay, settings.m_nbFECBlocks); - } - if ((m_settings.m_dataAddress != settings.m_dataAddress) || force) { m_dataAddress = settings.m_dataAddress; } @@ -243,11 +215,24 @@ void RemoteSinkSink::applySettings(const RemoteSinkSettings& settings, bool forc m_dataPort = settings.m_dataPort; } + if ((m_settings.m_log2Decim != settings.m_log2Decim) + || (m_settings.m_filterChainHash != settings.m_filterChainHash) + || (m_settings.m_nbFECBlocks != settings.m_nbFECBlocks) + || (m_settings.m_txDelay != settings.m_txDelay) || force) + { + double shiftFactor = HBFilterChainConverter::getShiftFactor(settings.m_log2Decim, settings.m_filterChainHash); + m_frequencyOffset = round(shiftFactor*m_basebandSampleRate); + setNbBlocksFEC(settings.m_nbFECBlocks); + setTxDelay(settings.m_txDelay, settings.m_nbFECBlocks, settings.m_log2Decim); + } + m_settings = settings; } -void RemoteSinkSink::applySampleRate(uint32_t sampleRate) +void RemoteSinkSink::applyBasebandSampleRate(uint32_t sampleRate) { - m_sampleRate = sampleRate; - setTxDelay(m_settings.m_txDelay, m_settings.m_nbFECBlocks); + m_basebandSampleRate = sampleRate; + double shiftFactor = HBFilterChainConverter::getShiftFactor(m_settings.m_log2Decim, m_settings.m_filterChainHash); + m_frequencyOffset = round(shiftFactor*m_basebandSampleRate); + setTxDelay(m_settings.m_txDelay, m_settings.m_nbFECBlocks, m_settings.m_log2Decim); } \ No newline at end of file diff --git a/plugins/channelrx/remotesink/remotesinksink.h b/plugins/channelrx/remotesink/remotesinksink.h index 58a14433c..49e278a50 100644 --- a/plugins/channelrx/remotesink/remotesinksink.h +++ b/plugins/channelrx/remotesink/remotesinksink.h @@ -19,7 +19,6 @@ #define INCLUDE_REMOTESINKSINK_H_ #include -#include #include "dsp/channelsamplesink.h" #include "channel/remotedatablock.h" @@ -28,7 +27,8 @@ #include "remotesinksettings.h" class DeviceSampleSource; -class RemoteSinkThread; +class RemoteSinkSender; +class QThread; class RemoteSinkSink : public QObject, public ChannelSampleSink { Q_OBJECT @@ -38,29 +38,26 @@ public: virtual void feed(const SampleVector::const_iterator& begin, const SampleVector::const_iterator& end); - void applySettings(const RemoteSinkSettings& settings, bool force = false); - void applySampleRate(uint32_t sampleRate); - void start(); - void stop(); - bool isRunning() const { return m_running; } + void startSender(); + void stopSender(); - /** Set center frequency given in Hz */ - void setCenterFrequency(uint64_t centerFrequency) { m_centerFrequency = centerFrequency; } + void applySettings(const RemoteSinkSettings& settings, bool force = false); + void applyBasebandSampleRate(uint32_t sampleRate); + + /** Set device center frequency given in Hz */ + void setDeviceCenterFrequency(uint64_t centerFrequency) { m_deviceCenterFrequency = centerFrequency; } /** Set sample rate given in Hz */ void setNbBlocksFEC(int nbBlocksFEC); - void setTxDelay(int txDelay, int nbBlocksFEC); + void setTxDelay(int txDelay, int nbBlocksFEC, int log2Decim); void setDataAddress(const QString& address) { m_dataAddress = address; } void setDataPort(uint16_t port) { m_dataPort = port; } -signals: - void dataBlockAvailable(RemoteDataBlock *dataBlock); - private: RemoteSinkSettings m_settings; - RemoteSinkThread *m_remoteSinkThread; - bool m_running; + QThread *m_senderThread; + RemoteSinkSender *m_remoteSinkSender; int m_txBlockIndex; //!< Current index in blocks to transmit in the Tx row uint16_t m_frameCount; //!< transmission frame count @@ -68,11 +65,10 @@ private: RemoteSuperBlock m_superBlock; RemoteMetaDataFEC m_currentMetaFEC; RemoteDataBlock *m_dataBlock; - QMutex m_dataBlockMutex; - uint64_t m_centerFrequency; + uint64_t m_deviceCenterFrequency; int64_t m_frequencyOffset; - uint32_t m_sampleRate; + uint32_t m_basebandSampleRate; int m_nbBlocksFEC; int m_txDelay; QString m_dataAddress;