mirror of
				https://github.com/f4exb/sdrangel.git
				synced 2025-10-30 12:30:20 -04:00 
			
		
		
		
	
		
			
	
	
		
			169 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			169 lines
		
	
	
		
			5.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|  | ///////////////////////////////////////////////////////////////////////////////////
 | ||
|  | // Copyright (C) 2023 Jon Beniston, M7RCE                                        //
 | ||
|  | //                                                                               //
 | ||
|  | // This program is free software; you can redistribute it and/or modify          //
 | ||
|  | // it under the terms of the GNU General Public License as published by          //
 | ||
|  | // the Free Software Foundation as version 3 of the License, or                  //
 | ||
|  | // (at your option) any later version.                                           //
 | ||
|  | //                                                                               //
 | ||
|  | // This program is distributed in the hope that it will be useful,               //
 | ||
|  | // but WITHOUT ANY WARRANTY; without even the implied warranty of                //
 | ||
|  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the                  //
 | ||
|  | // GNU General Public License V3 for more details.                               //
 | ||
|  | //                                                                               //
 | ||
|  | // You should have received a copy of the GNU General Public License             //
 | ||
|  | // along with this program. If not, see <http://www.gnu.org/licenses/>.          //
 | ||
|  | ///////////////////////////////////////////////////////////////////////////////////
 | ||
|  | 
 | ||
|  | #include <QDebug>
 | ||
|  | 
 | ||
|  | #include "util/morse.h"
 | ||
|  | 
 | ||
|  | #include "morsedemod.h"
 | ||
|  | 
 | ||
|  | MESSAGE_CLASS_DEFINITION(MorseDemod::MsgReportIdent, Message) | ||
|  | 
 | ||
|  | MorseDemod::MorseDemod() : | ||
|  |     m_movingAverageIdent(5000), | ||
|  |     m_prevBit(0), | ||
|  |     m_bitTime(0) | ||
|  | { | ||
|  | } | ||
|  | 
 | ||
|  | void MorseDemod::reset() | ||
|  | { | ||
|  |     m_binSampleCnt = 0; | ||
|  |     m_binCnt = 0; | ||
|  |     m_identNoise = 0.0001f; | ||
|  |     for (int i = 0; i < m_identBins; i++) | ||
|  |     { | ||
|  |         m_identMaxs[i] = 0.0f; | ||
|  |     } | ||
|  |     m_ident = ""; | ||
|  | } | ||
|  | 
 | ||
|  | void MorseDemod::applyChannelSettings(int channelSampleRate) | ||
|  | { | ||
|  |     if (channelSampleRate <= 0) { | ||
|  |         return; | ||
|  |     } | ||
|  |     m_samplesPerDot7wpm = channelSampleRate*60/(50*7); | ||
|  |     m_samplesPerDot10wpm = channelSampleRate*60/(50*10); | ||
|  | 
 | ||
|  |     m_ncoIdent.setFreq(-1020, channelSampleRate);  // +-50Hz source offset allowed
 | ||
|  |     m_bandpassIdent.create(1001, channelSampleRate, 970.0f, 1070.0f); // Ident at 1020
 | ||
|  | 
 | ||
|  |     m_lowpassIdent.create(301, channelSampleRate, 100.0f); | ||
|  |     m_movingAverageIdent.resize(m_samplesPerDot10wpm/5);  // Needs to be short enough for noise floor calculation
 | ||
|  | 
 | ||
|  |     reset(); | ||
|  | } | ||
|  | 
 | ||
|  | void MorseDemod::applySettings(int identThreshold) | ||
|  | { | ||
|  |     m_identThreshold = identThreshold; | ||
|  |     reset(); | ||
|  | } | ||
|  | 
 | ||
|  | void MorseDemod::processOneSample(const Complex &magc) | ||
|  | { | ||
|  |     // Filter to remove voice
 | ||
|  |     Complex c1 = m_bandpassIdent.filter(magc); | ||
|  |     // Remove ident sub-carrier offset
 | ||
|  |     c1 *= m_ncoIdent.nextIQ(); | ||
|  |     // Filter other signals
 | ||
|  |     Complex c2 = std::abs(m_lowpassIdent.filter(c1)); | ||
|  | 
 | ||
|  |     // Filter noise with moving average (moving average preserves edges)
 | ||
|  |     m_movingAverageIdent(c2.real()); | ||
|  |     Real mav = m_movingAverageIdent.asFloat(); | ||
|  | 
 | ||
|  |     // Caclulate noise floor
 | ||
|  |     if (mav > m_identMaxs[m_binCnt]) | ||
|  |         m_identMaxs[m_binCnt] = mav; | ||
|  |     m_binSampleCnt++; | ||
|  |     if (m_binSampleCnt >= m_samplesPerDot10wpm/4) | ||
|  |     { | ||
|  |         // Calc minimum of maximums
 | ||
|  |         m_identNoise = 1.0f; | ||
|  |         for (int i = 0; i < m_identBins; i++) | ||
|  |         { | ||
|  |             m_identNoise = std::min(m_identNoise, m_identMaxs[i]); | ||
|  |         } | ||
|  |         m_binSampleCnt = 0; | ||
|  |         m_binCnt++; | ||
|  |         if (m_binCnt == m_identBins) | ||
|  |             m_binCnt = 0; | ||
|  |         m_identMaxs[m_binCnt] = 0.0f; | ||
|  | 
 | ||
|  |         // Prevent divide by zero
 | ||
|  |         if (m_identNoise == 0.0f) | ||
|  |             m_identNoise = 1e-20f; | ||
|  |     } | ||
|  | 
 | ||
|  |     // CW demod
 | ||
|  |     int bit = (mav / m_identNoise) >= m_identThreshold; | ||
|  |     //m_stream << mav << "," << m_identNoise << "," << bit << "," << (mav / m_identNoise) << "\n";
 | ||
|  |     if ((m_prevBit == 0) && (bit == 1)) | ||
|  |     { | ||
|  |         if (m_bitTime > 7*m_samplesPerDot10wpm) | ||
|  |         { | ||
|  |             if (m_ident.trimmed().size() > 2) // Filter out noise that may appear as one or two characters
 | ||
|  |             { | ||
|  |                 qDebug() << "MorseDemod::processOneSample:" << m_ident << " " << Morse::toString(m_ident); | ||
|  | 
 | ||
|  |                 if (getMessageQueueToChannel()) | ||
|  |                 { | ||
|  |                     MorseDemod::MsgReportIdent *msg = MorseDemod::MsgReportIdent::create(m_ident); | ||
|  |                     getMessageQueueToChannel()->push(msg); | ||
|  |                 } | ||
|  |             } | ||
|  |             m_ident = ""; | ||
|  |         } | ||
|  |         else if (m_bitTime > 2.5*m_samplesPerDot10wpm) | ||
|  |         { | ||
|  |             m_ident.append(" "); | ||
|  |         } | ||
|  |         m_bitTime = 0; | ||
|  |     } | ||
|  |     else if (bit == 1) | ||
|  |     { | ||
|  |         m_bitTime++; | ||
|  |     } | ||
|  |     else if ((m_prevBit == 1) && (bit == 0)) | ||
|  |     { | ||
|  |         if (m_bitTime > 2*m_samplesPerDot10wpm) | ||
|  |         { | ||
|  |             m_ident.append("-"); | ||
|  |         } | ||
|  |         else if (m_bitTime > 0.2*m_samplesPerDot10wpm) | ||
|  |         { | ||
|  |             m_ident.append("."); | ||
|  |         } | ||
|  |         m_bitTime = 0; | ||
|  |     } | ||
|  |     else | ||
|  |     { | ||
|  |         m_bitTime++; | ||
|  |         if (m_bitTime > 10*m_samplesPerDot7wpm) | ||
|  |         { | ||
|  |             m_ident = m_ident.simplified(); | ||
|  |             if (m_ident.trimmed().size() > 2) // Filter out noise that may appear as one or two characters
 | ||
|  |                 { | ||
|  |                 qDebug() << "MorseDemod::processOneSample:" << m_ident << " " << Morse::toString(m_ident); | ||
|  | 
 | ||
|  |                 if (getMessageQueueToChannel()) | ||
|  |                 { | ||
|  |                     MorseDemod::MsgReportIdent *msg = MorseDemod::MsgReportIdent::create(m_ident); | ||
|  |                     getMessageQueueToChannel()->push(msg); | ||
|  |                 } | ||
|  | 
 | ||
|  |             } | ||
|  |             m_ident = ""; | ||
|  |             m_bitTime = 0; | ||
|  |         } | ||
|  |     } | ||
|  |     m_prevBit = bit; | ||
|  | } | ||
|  | 
 |