| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  | // Copyright (c) Charles J. Cliffe
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0+
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "ModemCW.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // We are given a baseband segment BW (default 500Hz) wide which we want to
 | 
					
						
							|  |  |  | // offset by mBeepFrequency (default 650Hz). This yields a spectrum.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //            |   |....|....|
 | 
					
						
							|  |  |  | //            |   |....|....|
 | 
					
						
							|  |  |  | //            |   |....|....|
 | 
					
						
							|  |  |  | // -----------|---|----|----|--
 | 
					
						
							|  |  |  | //            0  150  650  1150
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | ModemCW::ModemCW() | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  |         : ModemAnalog(), | 
					
						
							|  |  |  |           mBeepFrequency(650.0), | 
					
						
							|  |  |  |           mGain(15.0), | 
					
						
							|  |  |  |           mAutoGain(true), | 
					
						
							|  |  |  |           mLO(nullptr), | 
					
						
							|  |  |  |           mToReal(nullptr) { | 
					
						
							|  |  |  |     mLO = nco_crcf_create(LIQUID_NCO); | 
					
						
							|  |  |  |     mToReal = firhilbf_create(5, 60.0f); | 
					
						
							|  |  |  |     useSignalOutput(true); | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ModemCW::~ModemCW() { | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  |     if (mLO) | 
					
						
							|  |  |  |         nco_crcf_destroy(mLO); | 
					
						
							|  |  |  |     if (mToReal) | 
					
						
							|  |  |  |         firhilbf_destroy(mToReal); | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  | ModemArgInfoList ModemCW::getSettings() { | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  |     ModemArgInfoList args; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ModemArgInfo offsetArg; | 
					
						
							|  |  |  |     offsetArg.key = "offset"; | 
					
						
							|  |  |  |     offsetArg.name = "Frequency Offset"; | 
					
						
							|  |  |  |     offsetArg.value = std::to_string(mBeepFrequency); | 
					
						
							|  |  |  |     offsetArg.units = "Hz"; | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  |     offsetArg.description = "Frequency Offset / Beep frequency (200-1000Hz)"; | 
					
						
							| 
									
										
										
										
											2021-04-22 00:34:45 -04:00
										 |  |  |     offsetArg.type = ModemArgInfo::Type::FLOAT; | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  |     offsetArg.range = ModemRange(200.0, 1000.0); | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  |     args.push_back(offsetArg); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ModemArgInfo autoGain; | 
					
						
							|  |  |  |     autoGain.key = "auto"; | 
					
						
							|  |  |  |     autoGain.name = "Auto Gain"; | 
					
						
							|  |  |  |     autoGain.value = "on"; | 
					
						
							| 
									
										
										
										
											2021-04-22 00:34:45 -04:00
										 |  |  |     autoGain.type = ModemArgInfo::Type::STRING; | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  |     std::vector<std::string> autoOpts; | 
					
						
							|  |  |  |     autoOpts.push_back("on"); | 
					
						
							|  |  |  |     autoOpts.push_back("off"); | 
					
						
							|  |  |  |     autoGain.optionNames = autoOpts; | 
					
						
							|  |  |  |     autoGain.options = autoOpts; | 
					
						
							|  |  |  |     args.push_back(autoGain); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ModemArgInfo gain; | 
					
						
							|  |  |  |     gain.key = "gain"; | 
					
						
							|  |  |  |     gain.name = "Audio Gain"; | 
					
						
							|  |  |  |     gain.value = "15"; | 
					
						
							|  |  |  |     gain.units = "dB"; | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  |     gain.description = "Gain Setting (0-40dB)"; | 
					
						
							|  |  |  |     gain.range = ModemRange(0.0, 40.0); | 
					
						
							| 
									
										
										
										
											2021-04-22 00:34:45 -04:00
										 |  |  |     gain.type = ModemArgInfo::Type::FLOAT; | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  |     args.push_back(gain); | 
					
						
							|  |  |  |     return args; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  | void ModemCW::writeSetting(std::string setting, std::string value) { | 
					
						
							|  |  |  |     if (setting == "offset") { | 
					
						
							|  |  |  |         mBeepFrequency = std::stof(value); | 
					
						
							|  |  |  |         rebuildKit(); | 
					
						
							|  |  |  |     } else if (setting == "auto") { | 
					
						
							|  |  |  |         mAutoGain = (value == "on"); | 
					
						
							|  |  |  |     } else if (setting == "gain") { | 
					
						
							|  |  |  |         mGain = std::stof(value); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  | std::string ModemCW::readSetting(std::string setting) { | 
					
						
							|  |  |  |     if (setting == "offset") { | 
					
						
							|  |  |  |         return std::to_string(mBeepFrequency); | 
					
						
							|  |  |  |     } else if (setting == "auto") { | 
					
						
							|  |  |  |         return (mAutoGain) ? "on" : "off"; | 
					
						
							|  |  |  |     } else if (setting == "gain") { | 
					
						
							|  |  |  |         return std::to_string(mGain); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return ""; | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ModemBase *ModemCW::factory() { | 
					
						
							|  |  |  |     return new ModemCW; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::string ModemCW::getName() { | 
					
						
							|  |  |  |     return "CW"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  | int ModemCW::checkSampleRate(long long srate, int /* arate */) { | 
					
						
							|  |  |  |     if (srate < MIN_BANDWIDTH) | 
					
						
							|  |  |  |         return MIN_BANDWIDTH; | 
					
						
							| 
									
										
										
										
											2021-04-22 00:34:45 -04:00
										 |  |  |     return (int)srate; | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int ModemCW::getDefaultSampleRate() { | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  |     return MIN_BANDWIDTH; | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // The modem object is asked to make a "ModemKit" given the IQ sample rate
 | 
					
						
							|  |  |  | // and the audio sample rate. For the CW modem the IQ sample rate is small
 | 
					
						
							|  |  |  | // or narrow bandwidth. The demodulated sample rate must be fast enough to
 | 
					
						
							|  |  |  | // sample 200-1000Hz tones. If the IQ sample rate is less than 2000Hz then
 | 
					
						
							|  |  |  | // one doesn't have the bandwidth for these tones. So we need to interpolate
 | 
					
						
							|  |  |  | // the input IQ to audioOut, frequency shift, then pass the real part.
 | 
					
						
							|  |  |  | // Simple solution is just interpolate the IQ data to the audio sample rate.
 | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  | ModemKit *ModemCW::buildKit(long long sampleRate, int audioSampleRate) { | 
					
						
							| 
									
										
										
										
											2021-04-04 22:16:44 -04:00
										 |  |  |     auto *kit = new ModemKitCW(); | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  |     float As = 60.0f; | 
					
						
							|  |  |  |     double ratio = double(audioSampleRate) / double(sampleRate); | 
					
						
							|  |  |  |     kit->sampleRate = sampleRate; | 
					
						
							|  |  |  |     kit->audioSampleRate = audioSampleRate; | 
					
						
							|  |  |  |     kit->audioResampleRatio = ratio; | 
					
						
							| 
									
										
										
										
											2021-04-22 00:34:45 -04:00
										 |  |  |     kit->mInputResampler = msresamp_cccf_create((float)ratio, As); | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  |     return kit; | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  | void ModemCW::disposeKit(ModemKit *kit) { | 
					
						
							| 
									
										
										
										
											2021-04-04 22:16:44 -04:00
										 |  |  |     auto *cwkit = (ModemKitCW *) kit; | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  |     msresamp_cccf_destroy(cwkit->mInputResampler); | 
					
						
							|  |  |  |     delete kit; | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  | void ModemCW::initOutputBuffers(ModemKitAnalog *akit, ModemIQData *input) { | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  |     bufSize = input->data.size(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!bufSize) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     double audio_resample_ratio = akit->audioResampleRatio; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  |     size_t audio_out_size = (size_t) ceil((double) (bufSize) * audio_resample_ratio) + 512; | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Just make everything the audio out size
 | 
					
						
							|  |  |  |     if (mInput.size() != audio_out_size) { | 
					
						
							|  |  |  |         if (mInput.capacity() < audio_out_size) { | 
					
						
							|  |  |  |             mInput.reserve(audio_out_size); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         mInput.resize(audio_out_size); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ModemCW::demodulate(ModemKit *kit, ModemIQData *input, AudioThreadInput *audioOut) { | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  |     unsigned int outSize; | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  |     float lsb; | 
					
						
							|  |  |  |     liquid_float_complex sig; | 
					
						
							| 
									
										
										
										
											2021-04-04 22:16:44 -04:00
										 |  |  |     auto *cwkit = (ModemKitCW *) kit; | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     initOutputBuffers(cwkit, input); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!bufSize) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Interpolate IQ samples to full audio band. We need to be able to
 | 
					
						
							|  |  |  |     // sample at 2 times the desired beep frequency.
 | 
					
						
							| 
									
										
										
										
											2021-04-22 00:34:45 -04:00
										 |  |  |     msresamp_cccf_execute(cwkit->mInputResampler, &input->data[0], (unsigned int)bufSize, &mInput[0], &outSize); | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Make the shoe fit.
 | 
					
						
							|  |  |  |     if (demodOutputData.size() != outSize) { | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  |         demodOutputData.resize(outSize); | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Set the LO to the desired beep frequency.
 | 
					
						
							| 
									
										
										
										
											2021-04-22 00:34:45 -04:00
										 |  |  |     nco_crcf_set_frequency(mLO, 2.0f * (float)M_PI * mBeepFrequency / kit->audioSampleRate); | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Mix up from base band by beep frequency. Extract real part
 | 
					
						
							| 
									
										
										
										
											2021-03-19 22:42:42 -04:00
										 |  |  |     for (unsigned int i = 0; i < outSize; i++) { | 
					
						
							|  |  |  |         nco_crcf_mix_up(mLO, mInput[i], &sig); | 
					
						
							|  |  |  |         nco_crcf_step(mLO); | 
					
						
							|  |  |  |         firhilbf_c2r_execute(mToReal, sig, &lsb, &demodOutputData[i]); | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Determine gain automagically (if desired)
 | 
					
						
							|  |  |  |     if (mAutoGain) { | 
					
						
							|  |  |  |         aOutputCeilMA = aOutputCeilMA + (aOutputCeil - aOutputCeilMA) * 0.025f; | 
					
						
							|  |  |  |         aOutputCeilMAA = aOutputCeilMAA + (aOutputCeilMA - aOutputCeilMAA) * 0.025f; | 
					
						
							|  |  |  |         aOutputCeil = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (size_t i = 0; i < outSize; i++) { | 
					
						
							|  |  |  |             if (demodOutputData[i] > aOutputCeil) { | 
					
						
							|  |  |  |                 aOutputCeil = demodOutputData[i]; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-22 00:34:45 -04:00
										 |  |  |         mGain = 10.0f * std::log10(0.5f / aOutputCeilMAA); | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Apply gain to demodulated output data
 | 
					
						
							|  |  |  |     for (size_t i = 0; i < outSize; i++) { | 
					
						
							| 
									
										
										
										
											2021-04-22 00:34:45 -04:00
										 |  |  |         demodOutputData[i] *= std::pow(10.0f, mGain / 10.0f); | 
					
						
							| 
									
										
										
										
											2021-01-15 07:12:54 -08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     audioOut->channels = 1; | 
					
						
							|  |  |  |     audioOut->sampleRate = cwkit->audioSampleRate; | 
					
						
							|  |  |  |     audioOut->data.assign(demodOutputData.begin(), demodOutputData.end()); | 
					
						
							|  |  |  | } |