| 
									
										
										
										
											2024-05-15 11:38:25 +02:00
										 |  |  | ///////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // Copyright (C) 2024 Edouard Griffiths, F4EXB <f4exb06@gmail.com>               //
 | 
					
						
							|  |  |  | //                                                                               //
 | 
					
						
							|  |  |  | // This program is free software; you can redistribute it and/or modify          //
 | 
					
						
							|  |  |  | // it under the terms of the GNU General Public License as published by          //
 | 
					
						
							|  |  |  | // the Free Software Foundation as version 3 of the License, or                  //
 | 
					
						
							|  |  |  | // (at your option) any later version.                                           //
 | 
					
						
							|  |  |  | //                                                                               //
 | 
					
						
							|  |  |  | // This program is distributed in the hope that it will be useful,               //
 | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of                //
 | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the                  //
 | 
					
						
							|  |  |  | // GNU General Public License V3 for more details.                               //
 | 
					
						
							|  |  |  | //                                                                               //
 | 
					
						
							|  |  |  | // You should have received a copy of the GNU General Public License             //
 | 
					
						
							|  |  |  | // along with this program. If not, see <http://www.gnu.org/licenses/>.          //
 | 
					
						
							|  |  |  | ///////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef INCLUDE_FEATURE_MORSEDECODERWORKER_H_
 | 
					
						
							|  |  |  | #define INCLUDE_FEATURE_MORSEDECODERWORKER_H_
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <vector>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-17 18:24:36 +02:00
										 |  |  | #include "ggmorse/ggmorse.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-15 11:38:25 +02:00
										 |  |  | #include <QObject>
 | 
					
						
							|  |  |  | #include <QRecursiveMutex>
 | 
					
						
							|  |  |  | #include <QByteArray>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "dsp/dsptypes.h"
 | 
					
						
							|  |  |  | #include "dsp/datafifo.h"
 | 
					
						
							|  |  |  | #include "util/message.h"
 | 
					
						
							|  |  |  | #include "util/messagequeue.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "morsedecodersettings.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-17 18:51:13 +02:00
										 |  |  | class ScopeVis; | 
					
						
							| 
									
										
										
										
											2024-05-18 10:36:30 +02:00
										 |  |  | class QTimer; | 
					
						
							| 
									
										
										
										
											2024-05-17 18:51:13 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-15 11:38:25 +02:00
										 |  |  | class MorseDecoderWorker : public QObject { | 
					
						
							|  |  |  |     Q_OBJECT | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     class MsgConfigureMorseDecoderWorker : public Message { | 
					
						
							|  |  |  |         MESSAGE_CLASS_DECLARATION | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public: | 
					
						
							|  |  |  |         const MorseDecoderSettings& getSettings() const { return m_settings; } | 
					
						
							|  |  |  |         const QList<QString>& getSettingsKeys() const { return m_settingsKeys; } | 
					
						
							|  |  |  |         bool getForce() const { return m_force; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         static MsgConfigureMorseDecoderWorker* create(const MorseDecoderSettings& settings, const QList<QString>& settingsKeys, bool force) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             return new MsgConfigureMorseDecoderWorker(settings, settingsKeys, force); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private: | 
					
						
							|  |  |  |         MorseDecoderSettings m_settings; | 
					
						
							|  |  |  |         QList<QString> m_settingsKeys; | 
					
						
							|  |  |  |         bool m_force; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         MsgConfigureMorseDecoderWorker(const MorseDecoderSettings& settings, const QList<QString>& settingsKeys, bool force) : | 
					
						
							|  |  |  |             Message(), | 
					
						
							|  |  |  |             m_settings(settings), | 
					
						
							|  |  |  |             m_settingsKeys(settingsKeys), | 
					
						
							|  |  |  |             m_force(force) | 
					
						
							|  |  |  |         { } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-19 18:56:18 +02:00
										 |  |  |     class MsgConfigureSampleRate : public Message { | 
					
						
							|  |  |  |         MESSAGE_CLASS_DECLARATION | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public: | 
					
						
							|  |  |  |         int getSampleRate() const { return m_sampleRate; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         static MsgConfigureSampleRate* create(int sampleRate) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             return new MsgConfigureSampleRate(sampleRate); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private: | 
					
						
							|  |  |  |         int m_sampleRate; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         MsgConfigureSampleRate(int sampleRate) : | 
					
						
							|  |  |  |             Message(), | 
					
						
							|  |  |  |             m_sampleRate(sampleRate) | 
					
						
							|  |  |  |         { } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-15 11:38:25 +02:00
										 |  |  |     class MsgConnectFifo : public Message { | 
					
						
							|  |  |  |         MESSAGE_CLASS_DECLARATION | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public: | 
					
						
							|  |  |  |         DataFifo *getFifo() { return m_fifo; } | 
					
						
							|  |  |  |         bool getConnect() const { return m_connect; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         static MsgConnectFifo* create(DataFifo *fifo, bool connect) { | 
					
						
							|  |  |  |             return new MsgConnectFifo(fifo, connect); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     private: | 
					
						
							|  |  |  |         DataFifo *m_fifo; | 
					
						
							|  |  |  |         bool m_connect; | 
					
						
							|  |  |  |         MsgConnectFifo(DataFifo *fifo, bool connect) : | 
					
						
							|  |  |  |             Message(), | 
					
						
							|  |  |  |             m_fifo(fifo), | 
					
						
							|  |  |  |             m_connect(connect) | 
					
						
							|  |  |  |         { } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     MorseDecoderWorker(); | 
					
						
							|  |  |  | 	~MorseDecoderWorker(); | 
					
						
							|  |  |  |     void reset(); | 
					
						
							|  |  |  |     void startWork(); | 
					
						
							|  |  |  |     void stopWork(); | 
					
						
							|  |  |  |     MessageQueue *getInputMessageQueue() { return &m_inputMessageQueue; } | 
					
						
							|  |  |  |     void setMessageQueueToFeature(MessageQueue *messageQueue) { m_msgQueueToFeature = messageQueue; } | 
					
						
							| 
									
										
										
										
											2024-05-17 18:51:13 +02:00
										 |  |  |     void setScopeVis(ScopeVis* scopeVis) { m_scopeVis = scopeVis; } | 
					
						
							| 
									
										
										
										
											2024-05-15 11:38:25 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     void applySampleRate(int sampleRate); | 
					
						
							|  |  |  | 	void applySettings(const MorseDecoderSettings& settings, const QList<QString>& settingsKeys, bool force = false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     DataFifo *m_dataFifo; | 
					
						
							|  |  |  |     int m_channelSampleRate; | 
					
						
							|  |  |  |     int m_sinkSampleRate; | 
					
						
							|  |  |  | 	MessageQueue m_inputMessageQueue;  //!< Queue for asynchronous inbound communication
 | 
					
						
							|  |  |  |     MessageQueue *m_msgQueueToFeature; //!< Queue to report channel change to main feature object
 | 
					
						
							|  |  |  |     MorseDecoderSettings m_settings; | 
					
						
							|  |  |  | 	double m_magsq; | 
					
						
							| 
									
										
										
										
											2024-05-17 18:24:36 +02:00
										 |  |  |     QByteArray m_bytesBuffer; | 
					
						
							|  |  |  |     int m_bytesBufferSize; | 
					
						
							|  |  |  |     int m_bytesBufferCount; | 
					
						
							|  |  |  |     QByteArray m_convBuffer; | 
					
						
							| 
									
										
										
										
											2024-05-15 11:38:25 +02:00
										 |  |  |     QRecursiveMutex m_mutex; | 
					
						
							| 
									
										
										
										
											2024-05-17 18:24:36 +02:00
										 |  |  |     GGMorse::Parameters *m_ggMorseParameters; | 
					
						
							|  |  |  |     GGMorse *m_ggMorse; | 
					
						
							|  |  |  |     bool m_auto; | 
					
						
							|  |  |  |     float m_pitchHz; | 
					
						
							|  |  |  |     float m_speedWPM; | 
					
						
							| 
									
										
										
										
											2024-05-17 18:51:13 +02:00
										 |  |  |     ScopeVis* m_scopeVis; | 
					
						
							| 
									
										
										
										
											2024-05-18 10:36:30 +02:00
										 |  |  |     QTimer *m_pollTimer; | 
					
						
							| 
									
										
										
										
											2024-05-15 11:38:25 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     void feedPart( | 
					
						
							|  |  |  |         const QByteArray::const_iterator& begin, | 
					
						
							|  |  |  |         const QByteArray::const_iterator& end, | 
					
						
							|  |  |  |         DataFifo::DataType dataType | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool handleMessage(const Message& cmd); | 
					
						
							| 
									
										
										
										
											2024-05-20 22:46:39 +02:00
										 |  |  |     int processBuffer(QByteArray& bytesBuffer); //!< return the number of bytes left
 | 
					
						
							| 
									
										
										
										
											2024-05-17 18:24:36 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // inline void processSample(
 | 
					
						
							|  |  |  |     //     DataFifo::DataType dataType,
 | 
					
						
							|  |  |  |     //     const QByteArray::const_iterator& begin,
 | 
					
						
							|  |  |  |     //     int nbBytesPerSample,
 | 
					
						
							|  |  |  |     //     int i
 | 
					
						
							|  |  |  |     // )
 | 
					
						
							|  |  |  |     // {
 | 
					
						
							|  |  |  |     //     int16_t *s = (int16_t*) begin + (i*nbBytesPerSample);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     //     switch(dataType)
 | 
					
						
							|  |  |  |     //     {
 | 
					
						
							|  |  |  |     //         case DataFifo::DataTypeI16: {
 | 
					
						
							|  |  |  |     //             m_sampleBuffer[i] = *s;
 | 
					
						
							|  |  |  |     //         }
 | 
					
						
							|  |  |  |     //         break;
 | 
					
						
							|  |  |  |     //         case DataFifo::DataTypeCI16: {
 | 
					
						
							|  |  |  |     //             int32_t re = s[2*i];
 | 
					
						
							|  |  |  |     //             int32_t im = s[2*i+1];
 | 
					
						
							|  |  |  |     //             m_sampleBuffer[i] = (int16_t) ((re+im) / 2);
 | 
					
						
							|  |  |  |     //         }
 | 
					
						
							|  |  |  |     //         break;
 | 
					
						
							|  |  |  |     //     }
 | 
					
						
							|  |  |  |     // }
 | 
					
						
							| 
									
										
										
										
											2024-05-15 11:38:25 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | private slots: | 
					
						
							|  |  |  |     void handleInputMessages(); | 
					
						
							|  |  |  |     void handleData(); //!< Handle data when samples have to be processed
 | 
					
						
							| 
									
										
										
										
											2024-05-18 10:36:30 +02:00
										 |  |  |     void pollingTick(); | 
					
						
							| 
									
										
										
										
											2024-05-15 11:38:25 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif // INCLUDE_FEATURE_MORSEDECODERWORKER_H_
 |