| 
									
										
										
										
											2015-12-06 19:47:55 +01:00
										 |  |  | ///////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // Copyright (C) 2015 F4EXB                                                      //
 | 
					
						
							|  |  |  | // written by Edouard Griffiths                                                  //
 | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +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                  //
 | 
					
						
							|  |  |  | //                                                                               //
 | 
					
						
							|  |  |  | // 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/>.          //
 | 
					
						
							|  |  |  | ///////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-02 13:18:07 +02:00
										 |  |  | #include "../../channelrx/udpsrc/udpsrc.h"
 | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-02 21:52:39 +02:00
										 |  |  | #include <dsp/downchannelizer.h>
 | 
					
						
							| 
									
										
										
										
											2015-12-02 02:28:31 +01:00
										 |  |  | #include <QUdpSocket>
 | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | #include <QThread>
 | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | #include <QHostAddress>
 | 
					
						
							|  |  |  | #include "dsp/dspengine.h"
 | 
					
						
							| 
									
										
										
										
											2016-10-02 13:18:07 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "../../channelrx/udpsrc/udpsrcgui.h"
 | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | MESSAGE_CLASS_DEFINITION(UDPSrc::MsgUDPSrcConfigure, Message) | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | MESSAGE_CLASS_DEFINITION(UDPSrc::MsgUDPSrcConfigureImmediate, Message) | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | MESSAGE_CLASS_DEFINITION(UDPSrc::MsgUDPSrcSpectrum, Message) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-02 22:29:04 +02:00
										 |  |  | UDPSrc::UDPSrc(MessageQueue* uiMessageQueue, UDPSrcGUI* udpSrcGUI, BasebandSampleSink* spectrum) : | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 	m_settingsMutex(QMutex::Recursive), | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 	m_udpPort(9999), | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 	m_audioFifo(4, 24000), | 
					
						
							|  |  |  | 	m_audioActive(false), | 
					
						
							| 
									
										
										
										
											2015-12-05 11:49:24 +01:00
										 |  |  | 	m_audioStereo(false), | 
					
						
							| 
									
										
										
										
											2016-04-04 03:44:06 +02:00
										 |  |  | 	m_volume(20), | 
					
						
							|  |  |  | 	m_fmDeviation(2500) | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2015-11-19 08:55:58 +01:00
										 |  |  | 	setObjectName("UDPSrc"); | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 	m_udpBuffer = new UDPSink<Sample>(this, udpBLockSampleSize, m_udpPort); | 
					
						
							| 
									
										
										
										
											2016-04-05 17:02:24 +02:00
										 |  |  | 	m_udpBufferMono = new UDPSink<FixReal>(this, udpBLockSampleSize, m_udpPort); | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 	m_audioSocket = new QUdpSocket(this); | 
					
						
							| 
									
										
										
										
											2016-04-06 09:33:29 +02:00
										 |  |  | 	m_udpAudioBuf = new char[m_udpAudioPayloadSize]; | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	m_audioBuffer.resize(1<<9); | 
					
						
							|  |  |  | 	m_audioBufferFill = 0; | 
					
						
							| 
									
										
										
										
											2015-12-02 02:28:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 	m_inputSampleRate = 96000; | 
					
						
							| 
									
										
										
										
											2016-04-05 17:02:24 +02:00
										 |  |  | 	m_sampleFormat = FormatS16LE; | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 	m_outputSampleRate = 48000; | 
					
						
							|  |  |  | 	m_rfBandwidth = 32000; | 
					
						
							| 
									
										
										
										
											2015-12-04 13:47:43 +01:00
										 |  |  | 	m_audioPort = m_udpPort - 1; | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 	m_nco.setFreq(0, m_inputSampleRate); | 
					
						
							|  |  |  | 	m_interpolator.create(16, m_inputSampleRate, m_rfBandwidth / 2.0); | 
					
						
							|  |  |  | 	m_sampleDistanceRemain = m_inputSampleRate / m_outputSampleRate; | 
					
						
							|  |  |  | 	m_uiMessageQueue = uiMessageQueue; | 
					
						
							| 
									
										
										
										
											2015-11-19 08:55:58 +01:00
										 |  |  | 	m_udpSrcGUI = udpSrcGUI; | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 	m_spectrum = spectrum; | 
					
						
							|  |  |  | 	m_spectrumEnabled = false; | 
					
						
							|  |  |  | 	m_nextSSBId = 0; | 
					
						
							|  |  |  | 	m_nextS16leId = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	m_last = 0; | 
					
						
							|  |  |  | 	m_this = 0; | 
					
						
							|  |  |  | 	m_scale = 0; | 
					
						
							|  |  |  | 	m_boost = 0; | 
					
						
							|  |  |  | 	m_magsq = 0; | 
					
						
							| 
									
										
										
										
											2016-04-05 17:02:24 +02:00
										 |  |  | 	UDPFilter = new fftfilt(0.0, (m_rfBandwidth / 2.0) / m_outputSampleRate, udpBLockSampleSize * sizeof(Sample)); | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-04 03:44:06 +02:00
										 |  |  | 	m_phaseDiscri.setFMScaling((float) m_outputSampleRate / (2.0f * m_fmDeviation)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-04 13:47:43 +01:00
										 |  |  | 	if (m_audioSocket->bind(QHostAddress::LocalHost, m_audioPort)) | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2015-12-04 13:47:43 +01:00
										 |  |  | 		qDebug("UDPSrc::UDPSrc: bind audio socket to port %d", m_audioPort); | 
					
						
							| 
									
										
										
										
											2016-04-06 09:33:29 +02:00
										 |  |  | 		connect(m_audioSocket, SIGNAL(readyRead()), this, SLOT(audioReadyRead()), Qt::QueuedConnection); | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		qWarning("UDPSrc::UDPSrc: cannot bind audio port"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 	//DSPEngine::instance()->addAudioSink(&m_audioFifo);
 | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | UDPSrc::~UDPSrc() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 	delete m_audioSocket; | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 	delete m_udpBuffer; | 
					
						
							| 
									
										
										
										
											2016-04-05 17:02:24 +02:00
										 |  |  | 	delete m_udpBufferMono; | 
					
						
							| 
									
										
										
										
											2016-04-06 09:33:29 +02:00
										 |  |  | 	delete[] m_udpAudioBuf; | 
					
						
							| 
									
										
										
										
											2015-11-19 08:55:58 +01:00
										 |  |  | 	if (UDPFilter) delete UDPFilter; | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 	if (m_audioActive) DSPEngine::instance()->removeAudioSink(&m_audioFifo); | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | void UDPSrc::configure(MessageQueue* messageQueue, | 
					
						
							|  |  |  | 		SampleFormat sampleFormat, | 
					
						
							|  |  |  | 		Real outputSampleRate, | 
					
						
							|  |  |  | 		Real rfBandwidth, | 
					
						
							| 
									
										
										
										
											2016-04-04 03:44:06 +02:00
										 |  |  | 		int fmDeviation, | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 		QString& udpAddress, | 
					
						
							|  |  |  | 		int udpPort, | 
					
						
							| 
									
										
										
										
											2015-12-05 12:03:26 +01:00
										 |  |  | 		int audioPort) | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 	Message* cmd = MsgUDPSrcConfigure::create(sampleFormat, | 
					
						
							|  |  |  | 			outputSampleRate, | 
					
						
							|  |  |  | 			rfBandwidth, | 
					
						
							| 
									
										
										
										
											2016-04-04 03:44:06 +02:00
										 |  |  | 			fmDeviation, | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 			udpAddress, | 
					
						
							|  |  |  | 			udpPort, | 
					
						
							| 
									
										
										
										
											2015-12-05 12:03:26 +01:00
										 |  |  | 			audioPort); | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 	messageQueue->push(cmd); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UDPSrc::configureImmediate(MessageQueue* messageQueue, | 
					
						
							| 
									
										
										
										
											2015-12-05 12:03:26 +01:00
										 |  |  | 		bool audioActive, | 
					
						
							| 
									
										
										
										
											2015-12-05 12:23:55 +01:00
										 |  |  | 		bool audioStereo, | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 		int boost, | 
					
						
							|  |  |  | 		int volume) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Message* cmd = MsgUDPSrcConfigureImmediate::create( | 
					
						
							| 
									
										
										
										
											2015-12-05 12:03:26 +01:00
										 |  |  | 			audioActive, | 
					
						
							| 
									
										
										
										
											2015-12-05 12:23:55 +01:00
										 |  |  | 			audioStereo, | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 			boost, | 
					
						
							|  |  |  | 			volume); | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 	messageQueue->push(cmd); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UDPSrc::setSpectrum(MessageQueue* messageQueue, bool enabled) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Message* cmd = MsgUDPSrcSpectrum::create(enabled); | 
					
						
							|  |  |  | 	messageQueue->push(cmd); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UDPSrc::feed(const SampleVector::const_iterator& begin, const SampleVector::const_iterator& end, bool positiveOnly) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Complex ci; | 
					
						
							|  |  |  | 	fftfilt::cmplx* sideband; | 
					
						
							|  |  |  | 	Real l, r; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	m_sampleBuffer.clear(); | 
					
						
							|  |  |  | 	m_settingsMutex.lock(); | 
					
						
							|  |  |  | 	int rescale = (1 << m_boost); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 	for(SampleVector::const_iterator it = begin; it < end; ++it) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 		Complex c(it->real(), it->imag()); | 
					
						
							|  |  |  | 		c *= m_nco.nextIQ(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-10 01:53:32 +02:00
										 |  |  | 		if(m_interpolator.decimate(&m_sampleDistanceRemain, c, &ci)) | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 			m_magsq = ((ci.real()*ci.real() +  ci.imag()*ci.imag())*rescale*rescale) / (1<<30); | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 			Sample s(ci.real() * rescale, ci.imag() * rescale); | 
					
						
							|  |  |  | 			m_sampleBuffer.push_back(s); | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 			m_sampleDistanceRemain += m_inputSampleRate / m_outputSampleRate; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-05 17:02:24 +02:00
										 |  |  | 			if (m_sampleFormat == FormatLSB) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				int n_out = UDPFilter->runSSB(ci, &sideband, false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (n_out) | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					for (int i = 0; i < n_out; i++) | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						l = sideband[i].real(); | 
					
						
							|  |  |  | 						r = sideband[i].imag(); | 
					
						
							|  |  |  | 						m_udpBuffer->write(Sample(l, r)); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if (m_sampleFormat == FormatUSB) | 
					
						
							| 
									
										
										
										
											2015-12-02 02:28:31 +01:00
										 |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 				int n_out = UDPFilter->runSSB(ci, &sideband, true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (n_out) | 
					
						
							| 
									
										
										
										
											2015-12-02 02:28:31 +01:00
										 |  |  | 				{ | 
					
						
							| 
									
										
										
										
											2016-04-05 17:02:24 +02:00
										 |  |  | 					for (int i = 0; i < n_out; i++) | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 					{ | 
					
						
							| 
									
										
										
										
											2016-04-05 17:02:24 +02:00
										 |  |  | 						l = sideband[i].real(); | 
					
						
							|  |  |  | 						r = sideband[i].imag(); | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 						m_udpBuffer->write(Sample(l, r)); | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 			else if (m_sampleFormat == FormatNFM) | 
					
						
							| 
									
										
										
										
											2015-12-02 02:28:31 +01:00
										 |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2016-04-04 03:44:06 +02:00
										 |  |  | 				Real demod = 32768.0f * m_phaseDiscri.phaseDiscriminator(ci); | 
					
						
							| 
									
										
										
										
											2016-04-04 22:28:56 +02:00
										 |  |  | 				m_udpBuffer->write(Sample(demod, demod)); | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-04-05 17:02:24 +02:00
										 |  |  | 			else if (m_sampleFormat == FormatNFMMono) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				FixReal demod = (FixReal) (32768.0f * m_phaseDiscri.phaseDiscriminator(ci)); | 
					
						
							|  |  |  | 				m_udpBufferMono->write(demod); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else if (m_sampleFormat == FormatLSBMono) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				int n_out = UDPFilter->runSSB(ci, &sideband, false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (n_out) | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					for (int i = 0; i < n_out; i++) | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						l = (sideband[i].real() + sideband[i].imag()) * 0.7; | 
					
						
							|  |  |  | 						m_udpBufferMono->write(l); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else if (m_sampleFormat == FormatUSBMono) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				int n_out = UDPFilter->runSSB(ci, &sideband, true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (n_out) | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					for (int i = 0; i < n_out; i++) | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						l = (sideband[i].real() + sideband[i].imag()) * 0.7; | 
					
						
							|  |  |  | 						m_udpBufferMono->write(l); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else if (m_sampleFormat == FormatAMMono) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				FixReal demod = (FixReal) (32768.0f * sqrt(m_magsq)); | 
					
						
							|  |  |  | 				m_udpBufferMono->write(demod); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 			else // Raw I/Q samples
 | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				m_udpBuffer->write(s); | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-04 03:44:06 +02:00
										 |  |  | 	//qDebug() << "UDPSrc::feed: " << m_sampleBuffer.size() * 4;
 | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if((m_spectrum != 0) && (m_spectrumEnabled)) | 
					
						
							| 
									
										
										
										
											2015-12-02 02:28:31 +01:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 		m_spectrum->feed(m_sampleBuffer.begin(), m_sampleBuffer.end(), positiveOnly); | 
					
						
							| 
									
										
										
										
											2015-12-02 02:28:31 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	m_settingsMutex.unlock(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UDPSrc::start() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-04-04 03:44:06 +02:00
										 |  |  | 	m_phaseDiscri.reset(); | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void UDPSrc::stop() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool UDPSrc::handleMessage(const Message& cmd) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	qDebug() << "UDPSrc::handleMessage"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-02 21:52:39 +02:00
										 |  |  | 	if (DownChannelizer::MsgChannelizerNotification::match(cmd)) | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-10-02 21:52:39 +02:00
										 |  |  | 		DownChannelizer::MsgChannelizerNotification& notif = (DownChannelizer::MsgChannelizerNotification&) cmd; | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		m_settingsMutex.lock(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		m_inputSampleRate = notif.getSampleRate(); | 
					
						
							|  |  |  | 		m_nco.setFreq(-notif.getFrequencyOffset(), m_inputSampleRate); | 
					
						
							|  |  |  | 		m_interpolator.create(16, m_inputSampleRate, m_rfBandwidth / 2.0); | 
					
						
							|  |  |  | 		m_sampleDistanceRemain = m_inputSampleRate / m_outputSampleRate; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		m_settingsMutex.unlock(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-19 08:55:58 +01:00
										 |  |  | 		qDebug() << "UDPSrc::handleMessage: MsgChannelizerNotification: m_inputSampleRate: " << m_inputSampleRate | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 				<< " frequencyOffset: " << notif.getFrequencyOffset(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 	else if (MsgUDPSrcConfigureImmediate::match(cmd)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		MsgUDPSrcConfigureImmediate& cfg = (MsgUDPSrcConfigureImmediate&) cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		m_settingsMutex.lock(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-05 12:03:26 +01:00
										 |  |  | 		if (cfg.getAudioActive() != m_audioActive) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			m_audioActive = cfg.getAudioActive(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (m_audioActive) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				m_audioBufferFill = 0; | 
					
						
							|  |  |  | 				DSPEngine::instance()->addAudioSink(&m_audioFifo); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				DSPEngine::instance()->removeAudioSink(&m_audioFifo); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-05 12:23:55 +01:00
										 |  |  | 		if (cfg.getAudioStereo() != m_audioStereo) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			m_audioStereo = cfg.getAudioStereo(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 		if (cfg.getBoost() != m_boost) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			m_boost = cfg.getBoost(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (cfg.getVolume() != m_volume) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			m_volume = cfg.getVolume(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		m_settingsMutex.unlock(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		qDebug() << "UDPSrc::handleMessage: MsgUDPSrcConfigureImmediate: " | 
					
						
							| 
									
										
										
										
											2015-12-05 12:03:26 +01:00
										 |  |  | 				<< " m_audioActive: " << m_audioActive | 
					
						
							| 
									
										
										
										
											2015-12-05 12:23:55 +01:00
										 |  |  | 				<< " m_audioStereo: " << m_audioStereo | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 				<< " m_boost: " << m_boost | 
					
						
							|  |  |  | 				<< " m_volume: " << m_volume; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 	else if (MsgUDPSrcConfigure::match(cmd)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		MsgUDPSrcConfigure& cfg = (MsgUDPSrcConfigure&) cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		m_settingsMutex.lock(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		m_sampleFormat = cfg.getSampleFormat(); | 
					
						
							|  |  |  | 		m_outputSampleRate = cfg.getOutputSampleRate(); | 
					
						
							|  |  |  | 		m_rfBandwidth = cfg.getRFBandwidth(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 		if (cfg.getUDPAddress() != m_udpAddressStr) | 
					
						
							| 
									
										
										
										
											2015-12-02 02:28:31 +01:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 			m_udpAddressStr = cfg.getUDPAddress(); | 
					
						
							|  |  |  | 			m_udpBuffer->setAddress(m_udpAddressStr); | 
					
						
							| 
									
										
										
										
											2016-04-05 17:02:24 +02:00
										 |  |  | 			m_udpBufferMono->setAddress(m_udpAddressStr); | 
					
						
							| 
									
										
										
										
											2015-12-02 02:28:31 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-19 08:55:58 +01:00
										 |  |  | 		if (cfg.getUDPPort() != m_udpPort) | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2015-11-19 08:55:58 +01:00
										 |  |  | 			m_udpPort = cfg.getUDPPort(); | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 			m_udpBuffer->setPort(m_udpPort); | 
					
						
							| 
									
										
										
										
											2016-04-05 17:02:24 +02:00
										 |  |  | 			m_udpBufferMono->setPort(m_udpPort); | 
					
						
							| 
									
										
										
										
											2015-12-04 13:47:43 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-04 13:47:43 +01:00
										 |  |  | 		if (cfg.getAudioPort() != m_audioPort) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2015-12-05 03:36:31 +01:00
										 |  |  | 			m_audioPort = cfg.getAudioPort(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-04 08:54:47 +01:00
										 |  |  | 			disconnect(m_audioSocket, SIGNAL(readyRead()), this, SLOT(audioReadyRead())); | 
					
						
							|  |  |  | 			delete m_audioSocket; | 
					
						
							|  |  |  | 			m_audioSocket = new QUdpSocket(this); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-06 09:33:29 +02:00
										 |  |  | 			if (m_audioSocket->bind(QHostAddress::LocalHost, m_audioPort)) | 
					
						
							| 
									
										
										
										
											2015-12-04 08:54:47 +01:00
										 |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2016-04-06 09:33:29 +02:00
										 |  |  | 				connect(m_audioSocket, SIGNAL(readyRead()), this, SLOT(audioReadyRead()), Qt::QueuedConnection); | 
					
						
							|  |  |  | 				qDebug("UDPSrc::handleMessage: audio socket bound to port %d", m_audioPort); | 
					
						
							| 
									
										
										
										
											2015-12-04 08:54:47 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			else | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 			{ | 
					
						
							|  |  |  | 				qWarning("UDPSrc::handleMessage: cannot bind audio socket"); | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-04 03:44:06 +02:00
										 |  |  | 		if (cfg.getFMDeviation() != m_fmDeviation) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			m_fmDeviation = cfg.getFMDeviation(); | 
					
						
							|  |  |  | 			m_phaseDiscri.setFMScaling((float) m_outputSampleRate / (2.0f * m_fmDeviation)); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 		m_interpolator.create(16, m_inputSampleRate, m_rfBandwidth / 2.0); | 
					
						
							|  |  |  | 		m_sampleDistanceRemain = m_inputSampleRate / m_outputSampleRate; | 
					
						
							| 
									
										
										
										
											2016-04-05 17:02:24 +02:00
										 |  |  | 		UDPFilter->create_filter(0.0, (m_rfBandwidth / 2.0) / m_outputSampleRate); | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		m_settingsMutex.unlock(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 		qDebug() << "UDPSrc::handleMessage: MsgUDPSrcConfigure: m_sampleFormat: " << m_sampleFormat | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 				<< " m_outputSampleRate: " << m_outputSampleRate | 
					
						
							|  |  |  | 				<< " m_rfBandwidth: " << m_rfBandwidth | 
					
						
							| 
									
										
										
										
											2015-12-02 02:28:31 +01:00
										 |  |  | 				<< " m_boost: " << m_boost | 
					
						
							| 
									
										
										
										
											2016-04-03 11:29:11 +02:00
										 |  |  | 				<< " m_udpAddressStr: " << m_udpAddressStr | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 				<< " m_udpPort: " << m_udpPort | 
					
						
							| 
									
										
										
										
											2015-12-05 12:03:26 +01:00
										 |  |  | 				<< " m_audioPort: " << m_audioPort; | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else if (MsgUDPSrcSpectrum::match(cmd)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		MsgUDPSrcSpectrum& spc = (MsgUDPSrcSpectrum&) cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		m_spectrumEnabled = spc.getEnabled(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 		qDebug() << "UDPSrc::handleMessage: MsgUDPSrcSpectrum: m_spectrumEnabled: " << m_spectrumEnabled; | 
					
						
							| 
									
										
										
										
											2015-11-19 03:27:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		if(m_spectrum != 0) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		   return m_spectrum->handleMessage(cmd); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | void UDPSrc::audioReadyRead() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	while (m_audioSocket->hasPendingDatagrams()) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-04-06 09:33:29 +02:00
										 |  |  | 	    qint64 pendingDataSize = m_audioSocket->pendingDatagramSize(); | 
					
						
							|  |  |  | 	    qint64 udpReadBytes = m_audioSocket->readDatagram(m_udpAudioBuf, pendingDataSize, 0, 0); | 
					
						
							|  |  |  | 		//qDebug("UDPSrc::audioReadyRead: %lld", udpReadBytes);
 | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 		if (m_audioActive) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2015-12-05 11:49:24 +01:00
										 |  |  | 			if (m_audioStereo) | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2016-04-06 09:33:29 +02:00
										 |  |  | 				for (int i = 0; i < udpReadBytes - 3; i += 4) | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 				{ | 
					
						
							| 
									
										
										
										
											2016-04-06 09:33:29 +02:00
										 |  |  | 					qint16 l_sample = (qint16) *(&m_udpAudioBuf[i]); | 
					
						
							|  |  |  | 					qint16 r_sample = (qint16) *(&m_udpAudioBuf[i+2]); | 
					
						
							|  |  |  | 					m_audioBuffer[m_audioBufferFill].l  = l_sample * m_volume; | 
					
						
							|  |  |  | 					m_audioBuffer[m_audioBufferFill].r  = r_sample * m_volume; | 
					
						
							| 
									
										
										
										
											2015-12-05 11:49:24 +01:00
										 |  |  | 					++m_audioBufferFill; | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-05 11:49:24 +01:00
										 |  |  | 					if (m_audioBufferFill >= m_audioBuffer.size()) | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 					{ | 
					
						
							| 
									
										
										
										
											2015-12-05 11:49:24 +01:00
										 |  |  | 						uint res = m_audioFifo.write((const quint8*)&m_audioBuffer[0], m_audioBufferFill, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						if (res != m_audioBufferFill) | 
					
						
							|  |  |  | 						{ | 
					
						
							|  |  |  | 							qDebug("UDPSrc::audioReadyRead: (stereo) lost %u samples", m_audioBufferFill - res); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						m_audioBufferFill = 0; | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2015-12-05 11:49:24 +01:00
										 |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2016-04-06 09:33:29 +02:00
										 |  |  | 				for (int i = 0; i < udpReadBytes - 1; i += 2) | 
					
						
							| 
									
										
										
										
											2015-12-05 11:49:24 +01:00
										 |  |  | 				{ | 
					
						
							| 
									
										
										
										
											2016-04-06 09:33:29 +02:00
										 |  |  | 					qint16 sample = (qint16) *(&m_udpAudioBuf[i]); | 
					
						
							|  |  |  | 					m_audioBuffer[m_audioBufferFill].l  = sample * m_volume; | 
					
						
							|  |  |  | 					m_audioBuffer[m_audioBufferFill].r  = sample * m_volume; | 
					
						
							| 
									
										
										
										
											2015-12-05 11:49:24 +01:00
										 |  |  | 					++m_audioBufferFill; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					if (m_audioBufferFill >= m_audioBuffer.size()) | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						uint res = m_audioFifo.write((const quint8*)&m_audioBuffer[0], m_audioBufferFill, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						if (res != m_audioBufferFill) | 
					
						
							|  |  |  | 						{ | 
					
						
							|  |  |  | 							qDebug("UDPSrc::audioReadyRead: (mono) lost %u samples", m_audioBufferFill - res); | 
					
						
							|  |  |  | 						} | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-05 11:49:24 +01:00
										 |  |  | 						m_audioBufferFill = 0; | 
					
						
							|  |  |  | 					} | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 				} | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 			if (m_audioFifo.write((const quint8*)&m_audioBuffer[0], m_audioBufferFill, 0) != m_audioBufferFill) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				qDebug("UDPSrc::audioReadyRead: lost samples"); | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-04 03:46:51 +01:00
										 |  |  | 			m_audioBufferFill = 0; | 
					
						
							| 
									
										
										
										
											2015-12-03 04:02:21 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	//qDebug("UDPSrc::audioReadyRead: done");
 | 
					
						
							|  |  |  | } |