| 
									
										
										
										
											2020-11-10 16:38:12 +01:00
										 |  |  | ///////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							| 
									
										
										
										
											2023-11-18 06:28:24 +01:00
										 |  |  | // Copyright (C) 2012 maintech GmbH, Otto-Hahn-Str. 15, 97204 Hoechberg, Germany //
 | 
					
						
							|  |  |  | // written by Christian Daniel                                                   //
 | 
					
						
							|  |  |  | // Copyright (C) 2015-2021 Edouard Griffiths, F4EXB <f4exb06@gmail.com>          //
 | 
					
						
							|  |  |  | // Copyright (C) 2022 Jiří Pinkava <jiri.pinkava@rossum.ai>                      //
 | 
					
						
							| 
									
										
										
										
											2020-11-10 16:38:12 +01:00
										 |  |  | //                                                                               //
 | 
					
						
							|  |  |  | // 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 <http://www.gnu.org/licenses/>.          //
 | 
					
						
							|  |  |  | ///////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <QMutexLocker>
 | 
					
						
							|  |  |  | #include <QDebug>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "dsp/downchannelizer.h"
 | 
					
						
							|  |  |  | #include "dsp/basebandsamplesink.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-29 20:09:28 +02:00
										 |  |  | #include "dsp/scopevis.h"
 | 
					
						
							| 
									
										
										
										
											2024-05-12 21:53:09 +02:00
										 |  |  | #include "dsp/devicesamplesource.h"
 | 
					
						
							| 
									
										
										
										
											2020-11-10 16:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "interferometerbaseband.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | MESSAGE_CLASS_DEFINITION(InterferometerBaseband::MsgConfigureChannelizer, Message) | 
					
						
							|  |  |  | MESSAGE_CLASS_DEFINITION(InterferometerBaseband::MsgSignalNotification, Message) | 
					
						
							|  |  |  | MESSAGE_CLASS_DEFINITION(InterferometerBaseband::MsgConfigureCorrelation, Message) | 
					
						
							| 
									
										
										
										
											2024-05-12 21:53:09 +02:00
										 |  |  | MESSAGE_CLASS_DEFINITION(InterferometerBaseband::MsgConfigureLocalDeviceSampleSource, Message) | 
					
						
							| 
									
										
										
										
											2020-11-10 16:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | InterferometerBaseband::InterferometerBaseband(int fftSize) : | 
					
						
							|  |  |  |     m_correlator(fftSize), | 
					
						
							|  |  |  |     m_spectrumSink(nullptr), | 
					
						
							| 
									
										
										
										
											2024-05-12 21:53:09 +02:00
										 |  |  |     m_scopeSink(nullptr), | 
					
						
							|  |  |  |     m_localSampleSource(nullptr), | 
					
						
							|  |  |  |     m_play(false) | 
					
						
							| 
									
										
										
										
											2020-11-10 16:38:12 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     m_sampleMIFifo.init(2, 96000 * 8); | 
					
						
							|  |  |  |     m_vbegin.resize(2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (int i = 0; i < 2; i++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_sinks[i].setStreamIndex(i); | 
					
						
							|  |  |  |         m_channelizers[i] = new DownChannelizer(&m_sinks[i]); | 
					
						
							|  |  |  |         m_sizes[i] = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QObject::connect( | 
					
						
							|  |  |  |         &m_sampleMIFifo, | 
					
						
							|  |  |  |         &SampleMIFifo::dataSyncReady, | 
					
						
							|  |  |  |         this, | 
					
						
							|  |  |  |         &InterferometerBaseband::handleData, | 
					
						
							|  |  |  |         Qt::QueuedConnection | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     connect(&m_inputMessageQueue, SIGNAL(messageEnqueued()), this, SLOT(handleInputMessages())); | 
					
						
							|  |  |  |     m_lastStream = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | InterferometerBaseband::~InterferometerBaseband() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     for (int i = 0; i < 2; i++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         delete m_channelizers[i]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void InterferometerBaseband::reset() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QMutexLocker mutexLocker(&m_mutex); | 
					
						
							|  |  |  |     m_sampleMIFifo.reset(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (int i = 0; i < 2; i++) { | 
					
						
							|  |  |  |         m_sinks[i].reset(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void InterferometerBaseband::feed(const SampleVector::const_iterator& begin, const SampleVector::const_iterator& end, unsigned int streamIndex) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (streamIndex > 1) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (streamIndex == m_lastStream) { | 
					
						
							|  |  |  |         qWarning("InterferometerBaseband::feed: twice same stream in a row: %u", streamIndex); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_lastStream = streamIndex; | 
					
						
							|  |  |  |     m_vbegin[streamIndex] = begin; | 
					
						
							|  |  |  |     m_sizes[streamIndex] = end - begin; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (streamIndex == 1) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (m_sizes[0] != m_sizes[1]) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             qWarning("InterferometerBaseband::feed: unequal sizes: [0]: %d [1]: %d", m_sizes[0], m_sizes[1]); | 
					
						
							|  |  |  |             m_sampleMIFifo.writeSync(m_vbegin, std::min(m_sizes[0], m_sizes[1])); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             m_sampleMIFifo.writeSync(m_vbegin, m_sizes[0]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void InterferometerBaseband::handleData() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     QMutexLocker mutexLocker(&m_mutex); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const std::vector<SampleVector>& data = m_sampleMIFifo.getData(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned int ipart1begin; | 
					
						
							|  |  |  |     unsigned int ipart1end; | 
					
						
							|  |  |  |     unsigned int ipart2begin; | 
					
						
							|  |  |  |     unsigned int ipart2end; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while ((m_sampleMIFifo.fillSync() > 0) && (m_inputMessageQueue.size() == 0)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_sampleMIFifo.readSync(ipart1begin, ipart1end, ipart2begin, ipart2end); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (ipart1begin != ipart1end) { // first part of FIFO data
 | 
					
						
							|  |  |  |             processFifo(data, ipart1begin, ipart1end); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (ipart2begin != ipart2end) { // second part of FIFO data (used when block wraps around)
 | 
					
						
							|  |  |  |             processFifo(data, ipart2begin, ipart2end); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void InterferometerBaseband::processFifo(const std::vector<SampleVector>& data, unsigned int ibegin, unsigned int iend) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     for (unsigned int stream = 0; stream < 2; stream++) { | 
					
						
							|  |  |  |         m_channelizers[stream]->feed(data[stream].begin() + ibegin, data[stream].begin() + iend); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     run(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void InterferometerBaseband::run() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_correlator.performCorr(m_sinks[0].getData(), m_sinks[0].getSize(), m_sinks[1].getData(), m_sinks[1].getSize())) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-05-29 20:09:28 +02:00
										 |  |  |         if (m_scopeSink) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             std::vector<SampleVector::const_iterator> vbegin; | 
					
						
							|  |  |  |             vbegin.push_back(m_correlator.m_tcorr.begin()); | 
					
						
							|  |  |  |             m_scopeSink->feed(vbegin, m_correlator.m_processed); | 
					
						
							| 
									
										
										
										
											2020-11-10 16:38:12 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (m_spectrumSink) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if ((m_correlator.getCorrType() == InterferometerSettings::CorrelationFFT) | 
					
						
							| 
									
										
										
										
											2024-05-12 00:00:06 +02:00
										 |  |  |                 || (m_correlator.getCorrType() == InterferometerSettings::CorrelationIFFT) | 
					
						
							|  |  |  |                 || (m_correlator.getCorrType() == InterferometerSettings::CorrelationIFFT2) | 
					
						
							|  |  |  |                 || (m_correlator.getCorrType() == InterferometerSettings::CorrelationIFFTStar)) | 
					
						
							| 
									
										
										
										
											2020-11-10 16:38:12 +01:00
										 |  |  |             { | 
					
						
							|  |  |  |                 m_spectrumSink->feed(m_correlator.m_scorr.begin(), m_correlator.m_scorr.begin() + m_correlator.m_processed, false); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 m_spectrumSink->feed(m_correlator.m_tcorr.begin(), m_correlator.m_tcorr.begin() + m_correlator.m_processed, false); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2024-05-12 21:53:09 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (m_localSampleSource && m_play) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if ((m_correlator.getCorrType() == InterferometerSettings::CorrelationFFT) | 
					
						
							|  |  |  |                 || (m_correlator.getCorrType() == InterferometerSettings::CorrelationIFFT) | 
					
						
							|  |  |  |                 || (m_correlator.getCorrType() == InterferometerSettings::CorrelationIFFT2) | 
					
						
							|  |  |  |                 || (m_correlator.getCorrType() == InterferometerSettings::CorrelationIFFTStar)) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 m_localSampleSource->getSampleFifo()->write(m_correlator.m_scorr.begin(), m_correlator.m_scorr.begin() + m_correlator.m_processed); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 m_localSampleSource->getSampleFifo()->write(m_correlator.m_tcorr.begin(), m_correlator.m_tcorr.begin() + m_correlator.m_processed); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-11-10 16:38:12 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (int i = 0; i < 2; i++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         std::copy( | 
					
						
							|  |  |  |             m_sinks[i].getData().begin() + m_correlator.m_processed, | 
					
						
							|  |  |  |             m_sinks[i].getData().begin() + m_correlator.m_processed + m_correlator.m_remaining[i], | 
					
						
							|  |  |  |             m_sinks[i].getData().begin() | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         m_sinks[i].setDataStart(m_correlator.m_remaining[i]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void InterferometerBaseband::handleInputMessages() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Message* message; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while ((message = m_inputMessageQueue.pop()) != 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2024-05-12 21:53:09 +02:00
										 |  |  |         qDebug("InterferometerBaseband::handleInputMessage: %s", message->getIdentifier()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (handleMessage(*message)) { | 
					
						
							| 
									
										
										
										
											2020-11-10 16:38:12 +01:00
										 |  |  | 			delete message; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool InterferometerBaseband::handleMessage(const Message& cmd) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (MsgConfigureChannelizer::match(cmd)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QMutexLocker mutexLocker(&m_mutex); | 
					
						
							|  |  |  |         MsgConfigureChannelizer& cfg = (MsgConfigureChannelizer&) cmd; | 
					
						
							|  |  |  |         int log2Decim = cfg.getLog2Decim(); | 
					
						
							|  |  |  |         int filterChainHash = cfg.getFilterChainHash(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         qDebug() << "InterferometerBaseband::handleMessage: MsgConfigureChannelizer:" | 
					
						
							|  |  |  |                 << " log2Decim: " << log2Decim | 
					
						
							|  |  |  |                 << " filterChainHash: " << filterChainHash; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (int i = 0; i < 2; i++) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             m_channelizers[i]->setDecimation(log2Decim, filterChainHash); | 
					
						
							|  |  |  |             m_sinks[i].reset(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (MsgSignalNotification::match(cmd)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QMutexLocker mutexLocker(&m_mutex); | 
					
						
							|  |  |  |         MsgSignalNotification& cfg = (MsgSignalNotification&) cmd; | 
					
						
							|  |  |  |         int inputSampleRate = cfg.getInputSampleRate(); | 
					
						
							|  |  |  |         qint64 centerFrequency = cfg.getCenterFrequency(); | 
					
						
							|  |  |  |         int streamIndex = cfg.getStreamIndex(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         qDebug() << "InterferometerBaseband::handleMessage: MsgSignalNotification:" | 
					
						
							|  |  |  |                 << " inputSampleRate: " << inputSampleRate | 
					
						
							|  |  |  |                 << " centerFrequency: " << centerFrequency | 
					
						
							|  |  |  |                 << " streamIndex: " << streamIndex; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (streamIndex < 2) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             m_channelizers[streamIndex]->setBasebandSampleRate(inputSampleRate); | 
					
						
							|  |  |  |             m_sinks[streamIndex].reset(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (MsgConfigureCorrelation::match(cmd)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QMutexLocker mutexLocker(&m_mutex); | 
					
						
							|  |  |  |         MsgConfigureCorrelation& cfg = (MsgConfigureCorrelation&) cmd; | 
					
						
							|  |  |  |         InterferometerSettings::CorrelationType correlationType = cfg.getCorrelationType(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         qDebug() << "InterferometerBaseband::handleMessage: MsgConfigureCorrelation:" | 
					
						
							|  |  |  |                 << " correlationType: " << correlationType; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         m_correlator.setCorrType(correlationType); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-05-12 21:53:09 +02:00
										 |  |  |     else if (MsgConfigureLocalDeviceSampleSource::match(cmd)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QMutexLocker mutexLocker(&m_mutex); | 
					
						
							|  |  |  |         MsgConfigureLocalDeviceSampleSource& notif  = (MsgConfigureLocalDeviceSampleSource&) cmd; | 
					
						
							|  |  |  |         qDebug() << "InterferometerBaseband::handleMessage: MsgConfigureLocalDeviceSampleSource: " << notif.getDeviceSampleSource(); | 
					
						
							|  |  |  |         m_localSampleSource = notif.getDeviceSampleSource(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return  true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-10 16:38:12 +01:00
										 |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         qDebug("InterferometerBaseband::handleMessage: unhandled: %s", cmd.getIdentifier()); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void InterferometerBaseband::setBasebandSampleRate(unsigned int sampleRate) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     for (int istream = 0; istream < 2; istream++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_channelizers[istream]->setBasebandSampleRate(sampleRate); | 
					
						
							|  |  |  |         m_sinks[istream].reset(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |