| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  | ///////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // Copyright (C) 2017 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/>.          //
 | 
					
						
							|  |  |  | ///////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <QDebug>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "dsp/upchannelizer.h"
 | 
					
						
							| 
									
										
										
										
											2017-08-17 04:23:36 +02:00
										 |  |  | #include "util/db.h"
 | 
					
						
							| 
									
										
										
										
											2017-08-16 04:00:33 +02:00
										 |  |  | #include "udpsinkmsg.h"
 | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  | #include "udpsink.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | MESSAGE_CLASS_DEFINITION(UDPSink::MsgUDPSinkConfigure, Message) | 
					
						
							| 
									
										
										
										
											2017-08-14 16:09:56 +02:00
										 |  |  | MESSAGE_CLASS_DEFINITION(UDPSink::MsgUDPSinkSpectrum, Message) | 
					
						
							| 
									
										
										
										
											2017-08-18 17:51:11 +02:00
										 |  |  | MESSAGE_CLASS_DEFINITION(UDPSink::MsgResetReadIndex, Message) | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | UDPSink::UDPSink(MessageQueue* uiMessageQueue, UDPSinkGUI* udpSinkGUI, BasebandSampleSink* spectrum) : | 
					
						
							|  |  |  |     m_uiMessageQueue(uiMessageQueue), | 
					
						
							|  |  |  |     m_udpSinkGUI(udpSinkGUI), | 
					
						
							|  |  |  |     m_spectrum(spectrum), | 
					
						
							| 
									
										
										
										
											2017-08-16 22:49:19 +02:00
										 |  |  |     m_spectrumEnabled(false), | 
					
						
							| 
									
										
										
										
											2017-08-16 23:37:01 +02:00
										 |  |  |     m_spectrumChunkSize(2160), | 
					
						
							| 
									
										
										
										
											2017-08-16 22:49:19 +02:00
										 |  |  |     m_spectrumChunkCounter(0), | 
					
						
							| 
									
										
										
										
											2017-08-14 22:50:28 +02:00
										 |  |  |     m_magsq(1e-10), | 
					
						
							| 
									
										
										
										
											2017-08-17 04:23:36 +02:00
										 |  |  |     m_movingAverage(16, 1e-10), | 
					
						
							|  |  |  |     m_inMovingAverage(480, 1e-10), | 
					
						
							| 
									
										
										
										
											2017-08-16 16:09:35 +02:00
										 |  |  |     m_sampleRateSum(0), | 
					
						
							|  |  |  |     m_sampleRateAvgCounter(0), | 
					
						
							| 
									
										
										
										
											2017-08-16 23:37:01 +02:00
										 |  |  |     m_levelCalcCount(0), | 
					
						
							|  |  |  |     m_peakLevel(0.0f), | 
					
						
							|  |  |  |     m_levelSum(0.0f), | 
					
						
							|  |  |  |     m_levelNbSamples(480), | 
					
						
							| 
									
										
										
										
											2017-08-17 04:23:36 +02:00
										 |  |  |     m_squelchOpen(false), | 
					
						
							|  |  |  |     m_squelchOpenCount(0), | 
					
						
							|  |  |  |     m_squelchCloseCount(0), | 
					
						
							|  |  |  |     m_squelchThreshold(4800), | 
					
						
							| 
									
										
										
										
											2017-08-18 12:30:51 +02:00
										 |  |  |     m_modPhasor(0.0f), | 
					
						
							| 
									
										
										
										
											2017-08-19 03:42:56 +02:00
										 |  |  |     m_SSBFilterBufferIndex(0), | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  |     m_settingsMutex(QMutex::Recursive) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-08-14 10:59:05 +02:00
										 |  |  |     setObjectName("UDPSink"); | 
					
						
							| 
									
										
										
										
											2017-08-16 04:00:33 +02:00
										 |  |  |     m_udpHandler.setFeedbackMessageQueue(&m_inputMessageQueue); | 
					
						
							| 
									
										
										
										
											2017-08-19 03:42:56 +02:00
										 |  |  |     m_SSBFilter = new fftfilt(m_config.m_lowCutoff / m_config.m_inputSampleRate, m_config.m_rfBandwidth / m_config.m_inputSampleRate, m_ssbFftLen); | 
					
						
							|  |  |  |     m_SSBFilterBuffer = new Complex[m_ssbFftLen>>1]; // filter returns data exactly half of its size
 | 
					
						
							| 
									
										
										
										
											2017-08-14 10:59:05 +02:00
										 |  |  |     apply(true); | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | UDPSink::~UDPSink() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-08-19 03:42:56 +02:00
										 |  |  |     delete[] m_SSBFilterBuffer; | 
					
						
							|  |  |  |     delete m_SSBFilter; | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UDPSink::start() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-08-15 20:23:49 +02:00
										 |  |  |     m_udpHandler.start(); | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UDPSink::stop() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-08-15 20:23:49 +02:00
										 |  |  |     m_udpHandler.stop(); | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UDPSink::pull(Sample& sample) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-08-14 10:59:05 +02:00
										 |  |  |     if (m_running.m_channelMute) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         sample.m_real = 0.0f; | 
					
						
							|  |  |  |         sample.m_imag = 0.0f; | 
					
						
							| 
									
										
										
										
											2017-08-17 04:23:36 +02:00
										 |  |  |         initSquelch(false); | 
					
						
							| 
									
										
										
										
											2017-08-14 10:59:05 +02:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Complex ci; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_settingsMutex.lock(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m_interpolatorDistance > 1.0f) // decimate
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         modulateSample(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while (!m_interpolator.decimate(&m_interpolatorDistanceRemain, m_modSample, &ci)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             modulateSample(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (m_interpolator.interpolate(&m_interpolatorDistanceRemain, m_modSample, &ci)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             modulateSample(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_interpolatorDistanceRemain += m_interpolatorDistance; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ci *= m_carrierNco.nextIQ(); // shift to carrier frequency
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_settingsMutex.unlock(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-17 04:23:36 +02:00
										 |  |  |     double magsq = ci.real() * ci.real() + ci.imag() * ci.imag(); | 
					
						
							| 
									
										
										
										
											2017-08-14 22:50:28 +02:00
										 |  |  |     magsq /= (1<<30); | 
					
						
							|  |  |  |     m_movingAverage.feed(magsq); | 
					
						
							|  |  |  |     m_magsq = m_movingAverage.average(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-14 10:59:05 +02:00
										 |  |  |     sample.m_real = (FixReal) ci.real(); | 
					
						
							|  |  |  |     sample.m_imag = (FixReal) ci.imag(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UDPSink::modulateSample() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-08-18 12:30:51 +02:00
										 |  |  |     if (m_running.m_sampleFormat == FormatS16LE) // Linear I/Q transponding
 | 
					
						
							| 
									
										
										
										
											2017-08-15 20:23:49 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-08-18 12:30:51 +02:00
										 |  |  |         Sample s; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-15 20:23:49 +02:00
										 |  |  |         m_udpHandler.readSample(s); | 
					
						
							| 
									
										
										
										
											2017-08-17 04:23:36 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         uint64_t magsq = s.m_real * s.m_real + s.m_imag * s.m_imag; | 
					
						
							|  |  |  |         m_inMovingAverage.feed(magsq/1073741824.0); | 
					
						
							|  |  |  |         m_inMagsq = m_inMovingAverage.average(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         calculateSquelch(m_inMagsq); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (m_squelchOpen) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2017-08-25 15:29:45 +02:00
										 |  |  |             m_modSample.real(s.m_real * m_running.m_gainOut); | 
					
						
							|  |  |  |             m_modSample.imag(s.m_imag * m_running.m_gainOut); | 
					
						
							| 
									
										
										
										
											2017-08-17 04:23:36 +02:00
										 |  |  |             calculateLevel(m_modSample); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             m_modSample.real(0.0f); | 
					
						
							|  |  |  |             m_modSample.imag(0.0f); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-08-15 20:23:49 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-08-25 10:32:18 +02:00
										 |  |  |     else if (m_running.m_sampleFormat == FormatNFM) | 
					
						
							| 
									
										
										
										
											2017-08-18 12:30:51 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         FixReal t; | 
					
						
							| 
									
										
										
										
											2017-08-25 10:32:18 +02:00
										 |  |  |         readMonoSample(t); | 
					
						
							| 
									
										
										
										
											2017-08-18 12:30:51 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         m_inMovingAverage.feed((t*t)/1073741824.0); | 
					
						
							|  |  |  |         m_inMagsq = m_inMovingAverage.average(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         calculateSquelch(m_inMagsq); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (m_squelchOpen) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2017-08-19 01:46:52 +02:00
										 |  |  |             m_modPhasor += (m_running.m_fmDeviation / m_running.m_inputSampleRate) * (t / 32768.0f) * M_PI * 2.0f; | 
					
						
							| 
									
										
										
										
											2017-08-25 15:29:45 +02:00
										 |  |  |             m_modSample.real(cos(m_modPhasor) * 10362.2f * m_running.m_gainOut); | 
					
						
							|  |  |  |             m_modSample.imag(sin(m_modPhasor) * 10362.2f * m_running.m_gainOut); | 
					
						
							| 
									
										
										
										
											2017-08-18 22:15:04 +02:00
										 |  |  |             calculateLevel(m_modSample); | 
					
						
							| 
									
										
										
										
											2017-08-18 12:30:51 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             m_modSample.real(0.0f); | 
					
						
							|  |  |  |             m_modSample.imag(0.0f); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-08-25 10:32:18 +02:00
										 |  |  |     else if (m_running.m_sampleFormat == FormatAM) | 
					
						
							| 
									
										
										
										
											2017-08-19 01:46:52 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         FixReal t; | 
					
						
							| 
									
										
										
										
											2017-08-25 10:32:18 +02:00
										 |  |  |         readMonoSample(t); | 
					
						
							| 
									
										
										
										
											2017-08-19 01:46:52 +02:00
										 |  |  |         m_inMovingAverage.feed((t*t)/1073741824.0); | 
					
						
							|  |  |  |         m_inMagsq = m_inMovingAverage.average(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         calculateSquelch(m_inMagsq); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (m_squelchOpen) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2017-08-25 15:29:45 +02:00
										 |  |  |             m_modSample.real(((t / 32768.0f)*m_running.m_amModFactor*m_running.m_gainOut + 1.0f) * 16384.0f); // modulate and scale zero frequency carrier
 | 
					
						
							| 
									
										
										
										
											2017-08-19 01:46:52 +02:00
										 |  |  |             m_modSample.imag(0.0f); | 
					
						
							|  |  |  |             calculateLevel(m_modSample); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             m_modSample.real(0.0f); | 
					
						
							|  |  |  |             m_modSample.imag(0.0f); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-08-25 10:32:18 +02:00
										 |  |  |     else if ((m_running.m_sampleFormat == FormatLSB) || (m_running.m_sampleFormat == FormatUSB)) | 
					
						
							| 
									
										
										
										
											2017-08-19 03:42:56 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |         FixReal t; | 
					
						
							|  |  |  |         Complex c, ci; | 
					
						
							|  |  |  |         fftfilt::cmplx *filtered; | 
					
						
							|  |  |  |         int n_out = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-25 10:32:18 +02:00
										 |  |  |         readMonoSample(t); | 
					
						
							| 
									
										
										
										
											2017-08-19 03:42:56 +02:00
										 |  |  |         m_inMovingAverage.feed((t*t)/1073741824.0); | 
					
						
							|  |  |  |         m_inMagsq = m_inMovingAverage.average(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         calculateSquelch(m_inMagsq); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (m_squelchOpen) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2017-08-25 15:29:45 +02:00
										 |  |  |             ci.real((t / 32768.0f) * m_running.m_gainOut); | 
					
						
							| 
									
										
										
										
											2017-08-19 03:42:56 +02:00
										 |  |  |             ci.imag(0.0f); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-19 09:28:50 +02:00
										 |  |  |             n_out = m_SSBFilter->runSSB(ci, &filtered, (m_running.m_sampleFormat == FormatUSB)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (n_out > 0) | 
					
						
							| 
									
										
										
										
											2017-08-19 03:42:56 +02:00
										 |  |  |             { | 
					
						
							|  |  |  |                 memcpy((void *) m_SSBFilterBuffer, (const void *) filtered, n_out*sizeof(Complex)); | 
					
						
							|  |  |  |                 m_SSBFilterBufferIndex = 0; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             c = m_SSBFilterBuffer[m_SSBFilterBufferIndex]; | 
					
						
							|  |  |  |             m_modSample.real(m_SSBFilterBuffer[m_SSBFilterBufferIndex].real() * 32768.0f); | 
					
						
							|  |  |  |             m_modSample.imag(m_SSBFilterBuffer[m_SSBFilterBufferIndex].imag() * 32768.0f); | 
					
						
							|  |  |  |             m_SSBFilterBufferIndex++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             calculateLevel(m_modSample); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             m_modSample.real(0.0f); | 
					
						
							|  |  |  |             m_modSample.imag(0.0f); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-08-15 20:23:49 +02:00
										 |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_modSample.real(0.0f); | 
					
						
							|  |  |  |         m_modSample.imag(0.0f); | 
					
						
							| 
									
										
										
										
											2017-08-17 04:23:36 +02:00
										 |  |  |         initSquelch(false); | 
					
						
							| 
									
										
										
										
											2017-08-15 20:23:49 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-08-16 22:49:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (m_spectrum && m_spectrumEnabled && (m_spectrumChunkCounter < m_spectrumChunkSize - 1)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         Sample s; | 
					
						
							|  |  |  |         s.m_real = (FixReal) m_modSample.real(); | 
					
						
							|  |  |  |         s.m_imag = (FixReal) m_modSample.imag(); | 
					
						
							|  |  |  |         m_sampleBuffer.push_back(s); | 
					
						
							|  |  |  |         m_spectrumChunkCounter++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_spectrum->feed(m_sampleBuffer.begin(), m_sampleBuffer.end(), false); | 
					
						
							|  |  |  |         m_sampleBuffer.clear(); | 
					
						
							|  |  |  |         m_spectrumChunkCounter = 0; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-16 23:37:01 +02:00
										 |  |  | void UDPSink::calculateLevel(Real sample) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_levelCalcCount < m_levelNbSamples) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_peakLevel = std::max(std::fabs(m_peakLevel), sample); | 
					
						
							|  |  |  |         m_levelSum += sample * sample; | 
					
						
							|  |  |  |         m_levelCalcCount++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-08-17 04:23:36 +02:00
										 |  |  |         qreal rmsLevel = m_levelSum > 0.0 ? sqrt(m_levelSum / m_levelNbSamples) : 0.0; | 
					
						
							| 
									
										
										
										
											2017-08-16 23:37:01 +02:00
										 |  |  |         //qDebug("NFMMod::calculateLevel: %f %f", rmsLevel, m_peakLevel);
 | 
					
						
							|  |  |  |         emit levelChanged(rmsLevel, m_peakLevel, m_levelNbSamples); | 
					
						
							|  |  |  |         m_peakLevel = 0.0f; | 
					
						
							|  |  |  |         m_levelSum = 0.0f; | 
					
						
							|  |  |  |         m_levelCalcCount = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UDPSink::calculateLevel(Complex sample) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Real t = std::abs(sample); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m_levelCalcCount < m_levelNbSamples) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_peakLevel = std::max(std::fabs(m_peakLevel), t); | 
					
						
							|  |  |  |         m_levelSum += (t * t); | 
					
						
							|  |  |  |         m_levelCalcCount++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-08-17 04:23:36 +02:00
										 |  |  |         qreal rmsLevel = m_levelSum > 0.0 ? sqrt((m_levelSum/(1<<30)) / m_levelNbSamples) : 0.0; | 
					
						
							| 
									
										
										
										
											2017-08-16 23:37:01 +02:00
										 |  |  |         emit levelChanged(rmsLevel, m_peakLevel / 32768.0, m_levelNbSamples); | 
					
						
							|  |  |  |         m_peakLevel = 0.0f; | 
					
						
							|  |  |  |         m_levelSum = 0.0f; | 
					
						
							|  |  |  |         m_levelCalcCount = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  | bool UDPSink::handleMessage(const Message& cmd) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (UpChannelizer::MsgChannelizerNotification::match(cmd)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         UpChannelizer::MsgChannelizerNotification& notif = (UpChannelizer::MsgChannelizerNotification&) cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         m_config.m_basebandSampleRate = notif.getBasebandSampleRate(); | 
					
						
							|  |  |  |         m_config.m_outputSampleRate = notif.getSampleRate(); | 
					
						
							|  |  |  |         m_config.m_inputFrequencyOffset = notif.getFrequencyOffset(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-16 22:49:19 +02:00
										 |  |  |         apply(false); | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         qDebug() << "UDPSink::handleMessage: MsgChannelizerNotification:" | 
					
						
							|  |  |  |                 << " m_basebandSampleRate: " << m_config.m_basebandSampleRate | 
					
						
							|  |  |  |                 << " m_outputSampleRate: " << m_config.m_outputSampleRate | 
					
						
							|  |  |  |                 << " m_inputFrequencyOffset: " << m_config.m_inputFrequencyOffset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (MsgUDPSinkConfigure::match(cmd)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         MsgUDPSinkConfigure& cfg = (MsgUDPSinkConfigure&) cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         m_config.m_sampleFormat = cfg.getSampleFormat(); | 
					
						
							|  |  |  |         m_config.m_inputSampleRate = cfg.getInputSampleRate(); | 
					
						
							|  |  |  |         m_config.m_rfBandwidth = cfg.getRFBandwidth(); | 
					
						
							|  |  |  |         m_config.m_fmDeviation = cfg.getFMDeviation(); | 
					
						
							|  |  |  |         m_config.m_udpAddressStr = cfg.getUDPAddress(); | 
					
						
							|  |  |  |         m_config.m_udpPort = cfg.getUDPPort(); | 
					
						
							| 
									
										
										
										
											2017-08-14 10:59:05 +02:00
										 |  |  |         m_config.m_channelMute = cfg.getChannelMute(); | 
					
						
							| 
									
										
										
										
											2017-08-25 15:29:45 +02:00
										 |  |  |         m_config.m_gainIn = cfg.getGainIn(); | 
					
						
							|  |  |  |         m_config.m_gainOut = cfg.getGainOut(); | 
					
						
							| 
									
										
										
										
											2017-08-17 04:23:36 +02:00
										 |  |  |         m_config.m_squelch = CalcDb::powerFromdB(cfg.getSquelchDB()); | 
					
						
							| 
									
										
										
										
											2017-08-19 00:52:10 +02:00
										 |  |  |         m_config.m_squelchGate = cfg.getSquelchGate(); | 
					
						
							| 
									
										
										
										
											2017-08-17 20:23:17 +02:00
										 |  |  |         m_config.m_squelchEnabled = cfg.getSquelchEnabled(); | 
					
						
							| 
									
										
										
										
											2017-08-25 02:21:34 +02:00
										 |  |  |         m_config.m_autoRWBalance = cfg.getAutoRWBalance(); | 
					
						
							| 
									
										
										
										
											2017-08-25 09:48:07 +02:00
										 |  |  |         m_config.m_stereoInput = cfg.getStereoInput(); | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-16 09:53:23 +02:00
										 |  |  |         apply(cfg.getForce()); | 
					
						
							| 
									
										
										
										
											2017-08-15 20:23:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-14 10:59:05 +02:00
										 |  |  |         qDebug() << "UDPSink::handleMessage: MsgUDPSinkConfigure:" | 
					
						
							|  |  |  |                 << " m_sampleFormat: " << m_config.m_sampleFormat | 
					
						
							|  |  |  |                 << " m_inputSampleRate: " << m_config.m_inputSampleRate | 
					
						
							|  |  |  |                 << " m_rfBandwidth: " << m_config.m_rfBandwidth | 
					
						
							|  |  |  |                 << " m_fmDeviation: " << m_config.m_fmDeviation | 
					
						
							|  |  |  |                 << " m_udpAddressStr: " << m_config.m_udpAddressStr | 
					
						
							|  |  |  |                 << " m_udpPort: " << m_config.m_udpPort | 
					
						
							| 
									
										
										
										
											2017-08-17 00:16:17 +02:00
										 |  |  |                 << " m_channelMute: " << m_config.m_channelMute | 
					
						
							| 
									
										
										
										
											2017-08-25 15:29:45 +02:00
										 |  |  |                 << " m_gainIn: " << m_config.m_gainIn | 
					
						
							|  |  |  |                 << " m_gainOut: " << m_config.m_gainOut | 
					
						
							| 
									
										
										
										
											2017-08-17 04:23:36 +02:00
										 |  |  |                 << " squelchDB: " << cfg.getSquelchDB() | 
					
						
							| 
									
										
										
										
											2017-08-19 00:52:10 +02:00
										 |  |  |                 << " m_squelchGate: " << m_config.m_squelchGate | 
					
						
							| 
									
										
										
										
											2017-08-17 20:23:17 +02:00
										 |  |  |                 << " m_squelch: " << m_config.m_squelch | 
					
						
							| 
									
										
										
										
											2017-08-25 02:21:34 +02:00
										 |  |  |                 << " m_squelchEnabled: " << m_config.m_squelchEnabled | 
					
						
							| 
									
										
										
										
											2017-08-25 09:48:07 +02:00
										 |  |  |                 << " m_autoRWBalance: " << m_config.m_autoRWBalance | 
					
						
							|  |  |  |                 << " m_stereoInput: " << m_config.m_stereoInput; | 
					
						
							| 
									
										
										
										
											2017-08-14 10:59:05 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-08-16 04:00:33 +02:00
										 |  |  |     else if (UDPSinkMessages::MsgSampleRateCorrection::match(cmd)) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-08-16 11:35:47 +02:00
										 |  |  |         UDPSinkMessages::MsgSampleRateCorrection& cfg = (UDPSinkMessages::MsgSampleRateCorrection&) cmd; | 
					
						
							| 
									
										
										
										
											2017-08-16 22:49:19 +02:00
										 |  |  |         Real newSampleRate = m_actualInputSampleRate + cfg.getCorrectionFactor() * m_actualInputSampleRate; | 
					
						
							| 
									
										
										
										
											2017-08-16 16:09:35 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-16 22:49:19 +02:00
										 |  |  |         // exclude values too way out nominal sample rate (20%)
 | 
					
						
							|  |  |  |         if ((newSampleRate < m_running.m_inputSampleRate * 1.2) && (newSampleRate >  m_running.m_inputSampleRate * 0.8)) | 
					
						
							| 
									
										
										
										
											2017-08-16 16:09:35 +02:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2017-08-16 22:49:19 +02:00
										 |  |  |             m_actualInputSampleRate = newSampleRate; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if ((cfg.getRawDeltaRatio() > -0.05) || (cfg.getRawDeltaRatio() < 0.05)) | 
					
						
							| 
									
										
										
										
											2017-08-16 16:09:35 +02:00
										 |  |  |             { | 
					
						
							| 
									
										
										
										
											2017-08-16 22:49:19 +02:00
										 |  |  |                 if (m_sampleRateAvgCounter < m_sampleRateAverageItems) | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     m_sampleRateSum += m_actualInputSampleRate; | 
					
						
							|  |  |  |                     m_sampleRateAvgCounter++; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 m_sampleRateSum = 0.0; | 
					
						
							|  |  |  |                 m_sampleRateAvgCounter = 0; | 
					
						
							| 
									
										
										
										
											2017-08-16 16:09:35 +02:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-16 22:49:19 +02:00
										 |  |  |             if (m_sampleRateAvgCounter == m_sampleRateAverageItems) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 float avgRate = m_sampleRateSum / m_sampleRateAverageItems; | 
					
						
							|  |  |  |                 qDebug("UDPSink::handleMessage: MsgSampleRateCorrection: corr: %+.6f new rate: %.0f: avg rate: %.0f", | 
					
						
							|  |  |  |                         cfg.getCorrectionFactor(), | 
					
						
							|  |  |  |                         m_actualInputSampleRate, | 
					
						
							|  |  |  |                         avgRate); | 
					
						
							|  |  |  |                 m_actualInputSampleRate = avgRate; | 
					
						
							|  |  |  |                 m_sampleRateSum = 0.0; | 
					
						
							|  |  |  |                 m_sampleRateAvgCounter = 0; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2017-08-17 01:03:09 +02:00
										 |  |  | //            else
 | 
					
						
							|  |  |  | //            {
 | 
					
						
							|  |  |  | //                qDebug("UDPSink::handleMessage: MsgSampleRateCorrection: corr: %+.6f new rate: %.0f",
 | 
					
						
							|  |  |  | //                        cfg.getCorrectionFactor(),
 | 
					
						
							|  |  |  | //                        m_actualInputSampleRate);
 | 
					
						
							|  |  |  | //            }
 | 
					
						
							| 
									
										
										
										
											2017-08-16 22:49:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             m_settingsMutex.lock(); | 
					
						
							|  |  |  |             m_interpolatorDistanceRemain = 0; | 
					
						
							|  |  |  |             m_interpolatorConsumed = false; | 
					
						
							|  |  |  |             m_interpolatorDistance = (Real) m_actualInputSampleRate / (Real) m_config.m_outputSampleRate; | 
					
						
							|  |  |  |             //m_interpolator.create(48, m_actualInputSampleRate, m_config.m_rfBandwidth / 2.2, 3.0); // causes clicking: leaving at standard frequency
 | 
					
						
							|  |  |  |             m_settingsMutex.unlock(); | 
					
						
							| 
									
										
										
										
											2017-08-16 16:09:35 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-08-16 11:35:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-16 22:49:19 +02:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (MsgUDPSinkSpectrum::match(cmd)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         MsgUDPSinkSpectrum& spc = (MsgUDPSinkSpectrum&) cmd; | 
					
						
							|  |  |  |         m_spectrumEnabled = spc.getEnabled(); | 
					
						
							|  |  |  |         qDebug() << "UDPSink::handleMessage: MsgUDPSinkSpectrum: m_spectrumEnabled: " << m_spectrumEnabled; | 
					
						
							| 
									
										
										
										
											2017-08-16 11:35:47 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-16 04:00:33 +02:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-08-18 17:51:11 +02:00
										 |  |  |     else if (MsgResetReadIndex::match(cmd)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_settingsMutex.lock(); | 
					
						
							|  |  |  |         m_udpHandler.resetReadIndex(); | 
					
						
							|  |  |  |         m_settingsMutex.unlock(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         qDebug() << "UDPSink::handleMessage: MsgResetReadIndex"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  |     else | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-08-16 22:49:19 +02:00
										 |  |  |         if(m_spectrum != 0) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |            return m_spectrum->handleMessage(cmd); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UDPSink::configure(MessageQueue* messageQueue, | 
					
						
							|  |  |  |         SampleFormat sampleFormat, | 
					
						
							|  |  |  |         Real outputSampleRate, | 
					
						
							|  |  |  |         Real rfBandwidth, | 
					
						
							|  |  |  |         int fmDeviation, | 
					
						
							| 
									
										
										
										
											2017-08-19 01:46:52 +02:00
										 |  |  |         Real amModFactor, | 
					
						
							| 
									
										
										
										
											2017-08-25 11:26:59 +02:00
										 |  |  |         const QString& udpAddress, | 
					
						
							| 
									
										
										
										
											2017-08-14 10:59:05 +02:00
										 |  |  |         int udpPort, | 
					
						
							| 
									
										
										
										
											2017-08-16 09:53:23 +02:00
										 |  |  |         bool channelMute, | 
					
						
							| 
									
										
										
										
											2017-08-25 15:29:45 +02:00
										 |  |  |         Real gainIn, | 
					
						
							|  |  |  |         Real gainOut, | 
					
						
							| 
									
										
										
										
											2017-08-17 04:23:36 +02:00
										 |  |  |         Real squelchDB, | 
					
						
							| 
									
										
										
										
											2017-08-19 00:52:10 +02:00
										 |  |  |         Real squelchGate, | 
					
						
							| 
									
										
										
										
											2017-08-17 20:23:17 +02:00
										 |  |  |         bool squelchEnabled, | 
					
						
							| 
									
										
										
										
											2017-08-25 02:21:34 +02:00
										 |  |  |         bool autoRWBalance, | 
					
						
							| 
									
										
										
										
											2017-08-25 09:48:07 +02:00
										 |  |  |         bool stereoInput, | 
					
						
							| 
									
										
										
										
											2017-08-16 09:53:23 +02:00
										 |  |  |         bool force) | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     Message* cmd = MsgUDPSinkConfigure::create(sampleFormat, | 
					
						
							|  |  |  |             outputSampleRate, | 
					
						
							|  |  |  |             rfBandwidth, | 
					
						
							|  |  |  |             fmDeviation, | 
					
						
							| 
									
										
										
										
											2017-08-19 01:46:52 +02:00
										 |  |  |             amModFactor, | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  |             udpAddress, | 
					
						
							| 
									
										
										
										
											2017-08-14 10:59:05 +02:00
										 |  |  |             udpPort, | 
					
						
							| 
									
										
										
										
											2017-08-16 09:53:23 +02:00
										 |  |  |             channelMute, | 
					
						
							| 
									
										
										
										
											2017-08-25 15:29:45 +02:00
										 |  |  |             gainIn, | 
					
						
							|  |  |  |             gainOut, | 
					
						
							| 
									
										
										
										
											2017-08-17 04:23:36 +02:00
										 |  |  |             squelchDB, | 
					
						
							| 
									
										
										
										
											2017-08-19 00:52:10 +02:00
										 |  |  |             squelchGate, | 
					
						
							| 
									
										
										
										
											2017-08-17 20:23:17 +02:00
										 |  |  |             squelchEnabled, | 
					
						
							| 
									
										
										
										
											2017-08-25 02:21:34 +02:00
										 |  |  |             autoRWBalance, | 
					
						
							| 
									
										
										
										
											2017-08-25 09:48:07 +02:00
										 |  |  |             stereoInput, | 
					
						
							| 
									
										
										
										
											2017-08-16 09:53:23 +02:00
										 |  |  |             force); | 
					
						
							| 
									
										
										
										
											2017-08-14 01:39:26 +02:00
										 |  |  |     messageQueue->push(cmd); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-14 16:09:56 +02:00
										 |  |  | void UDPSink::setSpectrum(MessageQueue* messageQueue, bool enabled) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Message* cmd = MsgUDPSinkSpectrum::create(enabled); | 
					
						
							|  |  |  |     messageQueue->push(cmd); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-18 17:51:11 +02:00
										 |  |  | void UDPSink::resetReadIndex(MessageQueue* messageQueue) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Message* cmd = MsgResetReadIndex::create(); | 
					
						
							|  |  |  |     messageQueue->push(cmd); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-14 16:09:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-14 10:59:05 +02:00
										 |  |  | void UDPSink::apply(bool force) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if ((m_config.m_inputFrequencyOffset != m_running.m_inputFrequencyOffset) || | 
					
						
							|  |  |  |         (m_config.m_outputSampleRate != m_running.m_outputSampleRate) || force) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_settingsMutex.lock(); | 
					
						
							|  |  |  |         m_carrierNco.setFreq(m_config.m_inputFrequencyOffset, m_config.m_outputSampleRate); | 
					
						
							|  |  |  |         m_settingsMutex.unlock(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if((m_config.m_outputSampleRate != m_running.m_outputSampleRate) || | 
					
						
							|  |  |  |        (m_config.m_rfBandwidth != m_running.m_rfBandwidth) || | 
					
						
							|  |  |  |        (m_config.m_inputSampleRate != m_running.m_inputSampleRate) || force) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_settingsMutex.lock(); | 
					
						
							|  |  |  |         m_interpolatorDistanceRemain = 0; | 
					
						
							|  |  |  |         m_interpolatorConsumed = false; | 
					
						
							|  |  |  |         m_interpolatorDistance = (Real) m_config.m_inputSampleRate / (Real) m_config.m_outputSampleRate; | 
					
						
							|  |  |  |         m_interpolator.create(48, m_config.m_inputSampleRate, m_config.m_rfBandwidth / 2.2, 3.0); | 
					
						
							| 
									
										
										
										
											2017-08-16 11:35:47 +02:00
										 |  |  |         m_actualInputSampleRate = m_config.m_inputSampleRate; | 
					
						
							|  |  |  |         m_udpHandler.resetReadIndex(); | 
					
						
							| 
									
										
										
										
											2017-08-16 16:09:35 +02:00
										 |  |  |         m_sampleRateSum = 0.0; | 
					
						
							|  |  |  |         m_sampleRateAvgCounter = 0; | 
					
						
							| 
									
										
										
										
											2017-08-16 22:49:19 +02:00
										 |  |  |         m_spectrumChunkSize = m_config.m_inputSampleRate * 0.05; // 50 ms chunk
 | 
					
						
							|  |  |  |         m_spectrumChunkCounter = 0; | 
					
						
							| 
									
										
										
										
											2017-08-16 23:37:01 +02:00
										 |  |  |         m_levelNbSamples = m_config.m_inputSampleRate * 0.01; // every 10 ms
 | 
					
						
							|  |  |  |         m_levelCalcCount = 0; | 
					
						
							|  |  |  |         m_peakLevel = 0.0f; | 
					
						
							|  |  |  |         m_levelSum = 0.0f; | 
					
						
							| 
									
										
										
										
											2017-08-17 01:12:17 +02:00
										 |  |  |         m_udpHandler.resizeBuffer(m_config.m_inputSampleRate); | 
					
						
							| 
									
										
										
										
											2017-08-17 04:23:36 +02:00
										 |  |  |         m_inMovingAverage.resize(m_config.m_inputSampleRate * 0.01, 1e-10); // 10 ms
 | 
					
						
							| 
									
										
										
										
											2017-08-19 00:52:10 +02:00
										 |  |  |         m_squelchThreshold = m_config.m_inputSampleRate * m_config.m_squelchGate; | 
					
						
							|  |  |  |         initSquelch(m_squelchOpen); | 
					
						
							| 
									
										
										
										
											2017-08-19 03:42:56 +02:00
										 |  |  |         m_SSBFilter->create_filter(m_config.m_lowCutoff / m_config.m_inputSampleRate, m_config.m_rfBandwidth / m_config.m_inputSampleRate); | 
					
						
							| 
									
										
										
										
											2017-08-14 10:59:05 +02:00
										 |  |  |         m_settingsMutex.unlock(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-19 00:52:10 +02:00
										 |  |  |     if ((m_config.m_squelchGate != m_running.m_squelchGate) || force) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_squelchThreshold = m_config.m_outputSampleRate * m_config.m_squelchGate; | 
					
						
							|  |  |  |         initSquelch(m_squelchOpen); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-15 20:23:49 +02:00
										 |  |  |     if ((m_config.m_udpAddressStr != m_running.m_udpAddressStr) || | 
					
						
							|  |  |  |         (m_config.m_udpPort != m_running.m_udpPort) || force) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2017-08-16 22:49:19 +02:00
										 |  |  |         m_settingsMutex.lock(); | 
					
						
							| 
									
										
										
										
											2017-08-15 20:23:49 +02:00
										 |  |  |         m_udpHandler.configureUDPLink(m_config.m_udpAddressStr, m_config.m_udpPort); | 
					
						
							| 
									
										
										
										
											2017-08-16 22:49:19 +02:00
										 |  |  |         m_settingsMutex.unlock(); | 
					
						
							| 
									
										
										
										
											2017-08-15 20:23:49 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ((m_config.m_channelMute != m_running.m_channelMute) || force) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (!m_config.m_channelMute) { | 
					
						
							|  |  |  |             m_udpHandler.resetReadIndex(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-25 02:46:02 +02:00
										 |  |  |     if ((m_config.m_autoRWBalance != m_running.m_autoRWBalance) || force) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_settingsMutex.lock(); | 
					
						
							|  |  |  |         m_udpHandler.setAutoRWBalance(m_config.m_autoRWBalance); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!m_config.m_autoRWBalance) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             m_interpolatorDistanceRemain = 0; | 
					
						
							|  |  |  |             m_interpolatorConsumed = false; | 
					
						
							|  |  |  |             m_interpolatorDistance = (Real) m_config.m_inputSampleRate / (Real) m_config.m_outputSampleRate; | 
					
						
							|  |  |  |             m_interpolator.create(48, m_config.m_inputSampleRate, m_config.m_rfBandwidth / 2.2, 3.0); | 
					
						
							|  |  |  |             m_actualInputSampleRate = m_config.m_inputSampleRate; | 
					
						
							|  |  |  |             m_udpHandler.resetReadIndex(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         m_settingsMutex.unlock(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-14 10:59:05 +02:00
										 |  |  |     m_running = m_config; | 
					
						
							|  |  |  | } |