#pragma once #include #include #include #include #include #include #include #include #include #include #include #define FAST_ATAN2_COEF1 FL_M_PI / 4.0f #define FAST_ATAN2_COEF2 3.0f * FAST_ATAN2_COEF1 inline float fast_arctan2(float y, float x) { float abs_y = fabsf(y); float r, angle; if (x == 0.0f && y == 0.0f) { return 0.0f; } if (x>=0.0f) { r = (x - abs_y) / (x + abs_y); angle = FAST_ATAN2_COEF1 - FAST_ATAN2_COEF1 * r; } else { r = (x + abs_y) / (abs_y - x); angle = FAST_ATAN2_COEF2 - FAST_ATAN2_COEF1 * r; } if (y < 0.0f) { return -angle; } return angle; } namespace dsp { class FloatFMDemod : public generic_block { public: FloatFMDemod() {} FloatFMDemod(stream* in, float sampleRate, float deviation) { init(in, sampleRate, deviation); } void init(stream* in, float sampleRate, float deviation) { _in = in; _sampleRate = sampleRate; _deviation = deviation; phasorSpeed = (2 * FL_M_PI) / (_sampleRate / _deviation); generic_block::registerInput(_in); generic_block::registerOutput(&out); generic_block::_block_init = true; } void setInput(stream* in) { assert(generic_block::_block_init); std::lock_guard lck(generic_block::ctrlMtx); generic_block::tempStop(); generic_block::unregisterInput(_in); _in = in; generic_block::registerInput(_in); generic_block::tempStart(); } void setSampleRate(float sampleRate) { assert(generic_block::_block_init); std::lock_guard lck(generic_block::ctrlMtx); generic_block::tempStop(); _sampleRate = sampleRate; phasorSpeed = (2 * FL_M_PI) / (_sampleRate / _deviation); generic_block::tempStart(); } float getSampleRate() { assert(generic_block::_block_init); return _sampleRate; } void setDeviation(float deviation) { assert(generic_block::_block_init); std::lock_guard lck(generic_block::ctrlMtx); generic_block::tempStop(); _deviation = deviation; phasorSpeed = (2 * FL_M_PI) / (_sampleRate / _deviation); generic_block::tempStart(); } float getDeviation() { assert(generic_block::_block_init); return _deviation; } int run() { int count = _in->read(); if (count < 0) { return -1; } // This is somehow faster than volk... float diff, currentPhase; for (int i = 0; i < count; i++) { currentPhase = fast_arctan2(_in->readBuf[i].im, _in->readBuf[i].re); diff = currentPhase - phase; if (diff > 3.1415926535f) { diff -= 2 * 3.1415926535f; } else if (diff <= -3.1415926535f) { diff += 2 * 3.1415926535f; } out.writeBuf[i] = diff / phasorSpeed; phase = currentPhase; } _in->flush(); if (!out.swap(count)) { return -1; } return count; } stream out; private: float phase = 0; float phasorSpeed, _sampleRate, _deviation; stream* _in; }; class FMDemod : public generic_block { public: FMDemod() {} FMDemod(stream* in, float sampleRate, float deviation) { init(in, sampleRate, deviation); } void init(stream* in, float sampleRate, float deviation) { _in = in; _sampleRate = sampleRate; _deviation = deviation; phasorSpeed = (2 * FL_M_PI) / (_sampleRate / _deviation); generic_block::registerInput(_in); generic_block::registerOutput(&out); generic_block::_block_init = true; } void setInput(stream* in) { assert(generic_block::_block_init); std::lock_guard lck(generic_block::ctrlMtx); generic_block::tempStop(); generic_block::unregisterInput(_in); _in = in; generic_block::registerInput(_in); generic_block::tempStart(); } void setSampleRate(float sampleRate) { assert(generic_block::_block_init); std::lock_guard lck(generic_block::ctrlMtx); generic_block::tempStop(); _sampleRate = sampleRate; phasorSpeed = (2 * FL_M_PI) / (_sampleRate / _deviation); generic_block::tempStart(); } float getSampleRate() { assert(generic_block::_block_init); return _sampleRate; } void setDeviation(float deviation) { assert(generic_block::_block_init); _deviation = deviation; phasorSpeed = (2 * FL_M_PI) / (_sampleRate / _deviation); } float getDeviation() { assert(generic_block::_block_init); return _deviation; } int run() { int count = _in->read(); if (count < 0) { return -1; } // This is somehow faster than volk... float diff, currentPhase; for (int i = 0; i < count; i++) { currentPhase = fast_arctan2(_in->readBuf[i].im, _in->readBuf[i].re); diff = currentPhase - phase; if (diff > 3.1415926535f) { diff -= 2 * 3.1415926535f; } else if (diff <= -3.1415926535f) { diff += 2 * 3.1415926535f; } out.writeBuf[i].l = diff / phasorSpeed; out.writeBuf[i].r = diff / phasorSpeed; phase = currentPhase; } _in->flush(); if (!out.swap(count)) { return -1; } return count; } stream out; private: float phase = 0; float phasorSpeed, _sampleRate, _deviation; stream* _in; }; class AMDemod : public generic_block { public: AMDemod() {} AMDemod(stream* in) { init(in); } void init(stream* in) { _in = in; generic_block::registerInput(_in); generic_block::registerOutput(&out); generic_block::_block_init = true; } void setInput(stream* in) { assert(generic_block::_block_init); std::lock_guard lck(generic_block::ctrlMtx); generic_block::tempStop(); generic_block::unregisterInput(_in); _in = in; generic_block::registerInput(_in); generic_block::tempStart(); } int run() { int count = _in->read(); if (count < 0) { return -1; } volk_32fc_magnitude_32f(out.writeBuf, (lv_32fc_t*)_in->readBuf, count); _in->flush(); for (int i = 0; i < count; i++) { out.writeBuf[i] -= avg; avg += out.writeBuf[i] * 10e-4; } if (!out.swap(count)) { return -1; } return count; } stream out; private: stream* _in; float avg = 0; }; class SSBDemod : public generic_block { public: SSBDemod() {} SSBDemod(stream* in, float sampleRate, float bandWidth, int mode) { init(in, sampleRate, bandWidth, mode); } ~SSBDemod() { if (!generic_block::_block_init) { return; } generic_block::stop(); delete[] buffer; generic_block::_block_init = false; } enum { MODE_USB, MODE_LSB, MODE_DSB }; void init(stream* in, float sampleRate, float bandWidth, int mode) { _in = in; _sampleRate = sampleRate; _bandWidth = bandWidth; _mode = mode; phase = lv_cmake(1.0f, 0.0f); switch (_mode) { case MODE_USB: phaseDelta = lv_cmake(std::cos((_bandWidth / _sampleRate) * FL_M_PI), std::sin((_bandWidth / _sampleRate) * FL_M_PI)); break; case MODE_LSB: phaseDelta = lv_cmake(std::cos(-(_bandWidth / _sampleRate) * FL_M_PI), std::sin(-(_bandWidth / _sampleRate) * FL_M_PI)); break; case MODE_DSB: phaseDelta = lv_cmake(1.0f, 0.0f); break; } buffer = new lv_32fc_t[STREAM_BUFFER_SIZE]; generic_block::registerInput(_in); generic_block::registerOutput(&out); generic_block::_block_init = true; } void setInput(stream* in) { assert(generic_block::_block_init); std::lock_guard lck(generic_block::ctrlMtx); generic_block::tempStop(); generic_block::unregisterInput(_in); _in = in; generic_block::registerInput(_in); generic_block::tempStart(); } void setSampleRate(float sampleRate) { assert(generic_block::_block_init); _sampleRate = sampleRate; switch (_mode) { case MODE_USB: phaseDelta = lv_cmake(std::cos((_bandWidth / _sampleRate) * FL_M_PI), std::sin((_bandWidth / _sampleRate) * FL_M_PI)); break; case MODE_LSB: phaseDelta = lv_cmake(std::cos(-(_bandWidth / _sampleRate) * FL_M_PI), std::sin(-(_bandWidth / _sampleRate) * FL_M_PI)); break; case MODE_DSB: phaseDelta = lv_cmake(1.0f, 0.0f); break; } } void setBandWidth(float bandWidth) { assert(generic_block::_block_init); _bandWidth = bandWidth; switch (_mode) { case MODE_USB: phaseDelta = lv_cmake(std::cos((_bandWidth / _sampleRate) * FL_M_PI), std::sin((_bandWidth / _sampleRate) * FL_M_PI)); break; case MODE_LSB: phaseDelta = lv_cmake(std::cos(-(_bandWidth / _sampleRate) * FL_M_PI), std::sin(-(_bandWidth / _sampleRate) * FL_M_PI)); break; case MODE_DSB: phaseDelta = lv_cmake(1.0f, 0.0f); break; } } void setMode(int mode) { assert(generic_block::_block_init); _mode = mode; switch (_mode) { case MODE_USB: phaseDelta = lv_cmake(std::cos((_bandWidth / _sampleRate) * FL_M_PI), std::sin((_bandWidth / _sampleRate) * FL_M_PI)); break; case MODE_LSB: phaseDelta = lv_cmake(std::cos(-(_bandWidth / _sampleRate) * FL_M_PI), std::sin(-(_bandWidth / _sampleRate) * FL_M_PI)); break; case MODE_DSB: phaseDelta = lv_cmake(1.0f, 0.0f); break; } } int run() { int count = _in->read(); if (count < 0) { return -1; } volk_32fc_s32fc_x2_rotator_32fc(buffer, (lv_32fc_t*)_in->readBuf, phaseDelta, &phase, count); volk_32fc_deinterleave_real_32f(out.writeBuf, buffer, count); _in->flush(); if (!out.swap(count)) { return -1; } return count; } stream out; private: int _mode; float _sampleRate, _bandWidth; stream* _in; lv_32fc_t* buffer; lv_32fc_t phase; lv_32fc_t phaseDelta; }; class MSKDemod : public generic_hier_block { public: MSKDemod() {} MSKDemod(stream* input, float sampleRate, float deviation, float baudRate, float omegaGain = (0.01*0.01) / 4, float muGain = 0.01f, float omegaRelLimit = 0.005f) { init(input, sampleRate, deviation, baudRate, omegaGain, muGain, omegaRelLimit); } void init(stream* input, float sampleRate, float deviation, float baudRate, float omegaGain = (0.01*0.01) / 4, float muGain = 0.01f, float omegaRelLimit = 0.005f) { _sampleRate = sampleRate; _deviation = deviation; _baudRate = baudRate; _omegaGain = omegaGain; _muGain = muGain; _omegaRelLimit = omegaRelLimit; demod.init(input, _sampleRate, _deviation); recov.init(&demod.out, _sampleRate / _baudRate, _omegaGain, _muGain, _omegaRelLimit); out = &recov.out; generic_hier_block::registerBlock(&demod); generic_hier_block::registerBlock(&recov); generic_hier_block::_block_init = true; } void setSampleRate(float sampleRate) { assert(generic_hier_block::_block_init); generic_hier_block::tempStop(); _sampleRate = sampleRate; demod.setSampleRate(_sampleRate); recov.setOmega(_sampleRate / _baudRate, _omegaRelLimit); generic_hier_block::tempStart(); } void setDeviation(float deviation) { assert(generic_hier_block::_block_init); _deviation = deviation; demod.setDeviation(deviation); } void setBaudRate(float baudRate, float omegaRelLimit) { assert(generic_hier_block::_block_init); _baudRate = baudRate; _omegaRelLimit = omegaRelLimit; recov.setOmega(_sampleRate / _baudRate, _omegaRelLimit); } void setMMGains(float omegaGain, float myGain) { assert(generic_hier_block::_block_init); _omegaGain = omegaGain; _muGain = myGain; recov.setGains(_omegaGain, _muGain); } void setOmegaRelLimit(float omegaRelLimit) { assert(generic_hier_block::_block_init); _omegaRelLimit = omegaRelLimit; recov.setOmegaRelLimit(_omegaRelLimit); } stream* out = NULL; private: FloatFMDemod demod; MMClockRecovery recov; float _sampleRate; float _deviation; float _baudRate; float _omegaGain; float _muGain; float _omegaRelLimit; }; template class PSKDemod : public generic_hier_block> { public: PSKDemod() {} PSKDemod(stream* input, float sampleRate, float baudRate, int RRCTapCount = 31, float RRCAlpha = 0.32f, float agcRate = 10e-4, float costasLoopBw = 0.004f, float omegaGain = (0.01*0.01) / 4, float muGain = 0.01f, float omegaRelLimit = 0.005f) { init(input, sampleRate, baudRate, RRCTapCount, RRCAlpha, agcRate, costasLoopBw, omegaGain, muGain, omegaRelLimit); } void init(stream* input, float sampleRate, float baudRate, int RRCTapCount = 31, float RRCAlpha = 0.32f, float agcRate = 10e-4, float costasLoopBw = 0.004f, float omegaGain = (0.01*0.01) / 4, float muGain = 0.01f, float omegaRelLimit = 0.005f) { _RRCTapCount = RRCTapCount; _RRCAlpha = RRCAlpha; _sampleRate = sampleRate; _agcRate = agcRate; _costasLoopBw = costasLoopBw; _baudRate = baudRate; _omegaGain = omegaGain; _muGain = muGain; _omegaRelLimit = omegaRelLimit; agc.init(input, 1.0f, 65535, _agcRate); taps.init(_RRCTapCount, _sampleRate, _baudRate, _RRCAlpha); rrc.init(&agc.out, &taps); demod.init(&rrc.out, _costasLoopBw); generic_hier_block>::registerBlock(&agc); generic_hier_block>::registerBlock(&rrc); generic_hier_block>::registerBlock(&demod); if constexpr (OFFSET) { delay.init(&demod.out); recov.init(&delay.out, _sampleRate / _baudRate, _omegaGain, _muGain, _omegaRelLimit); generic_hier_block>::registerBlock(&delay); } else { recov.init(&demod.out, _sampleRate / _baudRate, _omegaGain, _muGain, _omegaRelLimit); } generic_hier_block>::registerBlock(&recov); out = &recov.out; generic_hier_block>::_block_init = true; } void setInput(stream* input) { assert((generic_hier_block>::_block_init)); agc.setInput(input); } void setSampleRate(float sampleRate) { assert((generic_hier_block>::_block_init)); _sampleRate = sampleRate; rrc.tempStop(); recov.tempStop(); taps.setSampleRate(_sampleRate); rrc.updateWindow(&taps); recov.setOmega(_sampleRate / _baudRate, _omegaRelLimit); rrc.tempStart(); recov.tempStart(); } void setBaudRate(float baudRate) { assert((generic_hier_block>::_block_init)); _baudRate = baudRate; rrc.tempStop(); recov.tempStop(); taps.setBaudRate(_baudRate); rrc.updateWindow(&taps); recov.setOmega(_sampleRate / _baudRate, _omegaRelLimit); rrc.tempStart(); recov.tempStart(); } void setRRCParams(int RRCTapCount, float RRCAlpha) { assert((generic_hier_block>::_block_init)); _RRCTapCount = RRCTapCount; _RRCAlpha = RRCAlpha; taps.setTapCount(_RRCTapCount); taps.setAlpha(RRCAlpha); rrc.updateWindow(&taps); } void setAgcRate(float agcRate) { assert((generic_hier_block>::_block_init)); _agcRate = agcRate; agc.setRate(_agcRate); } void setCostasLoopBw(float costasLoopBw) { assert((generic_hier_block>::_block_init)); _costasLoopBw = costasLoopBw; demod.setLoopBandwidth(_costasLoopBw); } void setMMGains(float omegaGain, float myGain) { assert((generic_hier_block>::_block_init)); _omegaGain = omegaGain; _muGain = myGain; recov.setGains(_omegaGain, _muGain); } void setOmegaRelLimit(float omegaRelLimit) { assert((generic_hier_block>::_block_init)); _omegaRelLimit = omegaRelLimit; recov.setOmegaRelLimit(_omegaRelLimit); } stream* out = NULL; private: dsp::ComplexAGC agc; dsp::RRCTaps taps; dsp::FIR rrc; CostasLoop demod; DelayImag delay; MMClockRecovery recov; int _RRCTapCount; float _RRCAlpha; float _sampleRate; float _agcRate; float _baudRate; float _costasLoopBw; float _omegaGain; float _muGain; float _omegaRelLimit; }; class PMDemod : public generic_hier_block { public: PMDemod() {} PMDemod(stream* input, float sampleRate, float baudRate, float agcRate = 0.02e-3f, float pllLoopBandwidth = (0.06f*0.06f) / 4.0f, int rrcTapCount = 31, float rrcAlpha = 0.6f, float omegaGain = (0.01*0.01) / 4, float muGain = 0.01f, float omegaRelLimit = 0.005f) { init(input, sampleRate, baudRate, agcRate, pllLoopBandwidth, rrcTapCount, rrcAlpha, omegaGain, muGain, omegaRelLimit); } void init(stream* input, float sampleRate, float baudRate, float agcRate = 0.02e-3f, float pllLoopBandwidth = (0.06f*0.06f) / 4.0f, int rrcTapCount = 31, float rrcAlpha = 0.6f, float omegaGain = (0.01*0.01) / 4, float muGain = 0.01f, float omegaRelLimit = 0.005f) { _sampleRate = sampleRate; _baudRate = baudRate; _agcRate = agcRate; _pllLoopBandwidth = pllLoopBandwidth; _rrcTapCount = rrcTapCount; _rrcAlpha = rrcAlpha; _omegaGain = omegaGain; _muGain = muGain; _omegaRelLimit = omegaRelLimit; agc.init(input, 1.0f, 65535, _agcRate); pll.init(&agc.out, _pllLoopBandwidth); rrcwin.init(_rrcTapCount, _sampleRate, _baudRate, _rrcAlpha); rrc.init(&pll.out, &rrcwin); recov.init(&rrc.out, _sampleRate / _baudRate, _omegaGain, _muGain, _omegaRelLimit); out = &recov.out; generic_hier_block::registerBlock(&agc); generic_hier_block::registerBlock(&pll); generic_hier_block::registerBlock(&rrc); generic_hier_block::registerBlock(&recov); generic_hier_block::_block_init = true; } void setInput(stream* input) { assert(generic_hier_block::_block_init); agc.setInput(input); } void setAgcRate(float agcRate) { assert(generic_hier_block::_block_init); _agcRate = agcRate; agc.setRate(_agcRate); } void setPllLoopBandwidth(float pllLoopBandwidth) { assert(generic_hier_block::_block_init); _pllLoopBandwidth = pllLoopBandwidth; pll.setLoopBandwidth(_pllLoopBandwidth); } void setRRCParams(int rrcTapCount, float rrcAlpha) { assert(generic_hier_block::_block_init); _rrcTapCount = rrcTapCount; _rrcAlpha = rrcAlpha; rrcwin.setTapCount(_rrcTapCount); rrcwin.setAlpha(_rrcAlpha); rrc.updateWindow(&rrcwin); } void setMMGains(float omegaGain, float muGain) { assert(generic_hier_block::_block_init); _omegaGain = omegaGain; _muGain = muGain; recov.setGains(_omegaGain, _muGain); } void setOmegaRelLimit(float omegaRelLimit) { assert(generic_hier_block::_block_init); _omegaRelLimit = omegaRelLimit; recov.setOmegaRelLimit(_omegaRelLimit); } stream* out = NULL; private: dsp::ComplexAGC agc; dsp::CarrierTrackingPLL pll; dsp::RRCTaps rrcwin; dsp::FIR rrc; dsp::MMClockRecovery recov; float _sampleRate; float _baudRate; float _agcRate; float _pllLoopBandwidth; int _rrcTapCount; float _rrcAlpha; float _omegaGain; float _muGain; float _omegaRelLimit; }; class StereoFMDemod : public generic_hier_block { public: StereoFMDemod() {} StereoFMDemod(stream* input, float sampleRate, float baudRate, float agcRate = 0.02e-3f, float pllLoopBandwidth = (0.06f*0.06f) / 4.0f, int rrcTapCount = 31, float rrcAlpha = 0.6f, float omegaGain = (0.01*0.01) / 4, float muGain = 0.01f, float omegaRelLimit = 0.005f) { init(input, sampleRate); } void init(stream* input, float sampleRate) { _sampleRate = sampleRate; r2c.init(input); split.init(&r2c.out); split.bindStream(&APlusBIn); split.bindStream(&AMinusBIn); split.bindStream(&PilotIn); APlusBWin.init(0, 17000, 2500, _sampleRate); AMinusBWin.init(38000, 38000 + 17000, 2500, _sampleRate); PilotWin.init(18500, 19500, 1500, _sampleRate); APlusBFir.init(&APlusBIn, &APlusBWin); AMinusBFir.init(&AMinusBIn, &AMinusBWin); PilotFir.init(&PilotIn, &PilotWin); pll.init(&PilotFir.out, 0.1f); p2s.init(&pll.out); mixer.init(&AMinusBFir.out, &p2s.out); c2rAPlusB.init(&APlusBFir.out); c2rAMinusB.init(&mixer.out); APlusBSplit.init(&c2rAPlusB.out); AMinusBSplit.init(&c2rAMinusB.out); APlusBSplit.bindStream(&AdderAPlusB); APlusBSplit.bindStream(&SubtractorAPlusB); AMinusBSplit.bindStream(&AdderAMinusB); AMinusBSplit.bindStream(&SubtractorAMinusB); Adder.init(&AdderAPlusB, &AdderAMinusB); Subtractor.init(&SubtractorAPlusB, &SubtractorAMinusB); c2s.init(&Adder.out, &Subtractor.out); out = &c2s.out; generic_hier_block::registerBlock(&r2c); generic_hier_block::registerBlock(&split); generic_hier_block::registerBlock(&APlusBFir); generic_hier_block::registerBlock(&AMinusBFir); generic_hier_block::registerBlock(&PilotFir); generic_hier_block::registerBlock(&pll); generic_hier_block::registerBlock(&p2s); generic_hier_block::registerBlock(&mixer); generic_hier_block::registerBlock(&c2rAPlusB); generic_hier_block::registerBlock(&c2rAMinusB); generic_hier_block::registerBlock(&APlusBSplit); generic_hier_block::registerBlock(&AMinusBSplit); generic_hier_block::registerBlock(&Adder); generic_hier_block::registerBlock(&Subtractor); generic_hier_block::registerBlock(&c2s); generic_hier_block::_block_init = true; } void setInput(stream* input) { assert(generic_hier_block::_block_init); r2c.setInput(input); } stream* out = NULL; private: filter_window::BandPassBlackmanWindow APlusBWin; filter_window::BandPassBlackmanWindow AMinusBWin; filter_window::BandPassBlackmanWindow PilotWin; RealToComplex r2c; Splitter split; stream APlusBIn; stream AMinusBIn; stream PilotIn; ComplexFIR APlusBFir; ComplexFIR AMinusBFir; ComplexFIR PilotFir; PLL pll; BFMPilotToStereo p2s; Multiply mixer; ComplexToReal c2rAPlusB; ComplexToReal c2rAMinusB; Splitter APlusBSplit; Splitter AMinusBSplit; stream AdderAPlusB; stream AdderAMinusB; stream SubtractorAPlusB; stream SubtractorAMinusB; Add Adder; Add Subtractor; ChannelsToStereo c2s; float _sampleRate; }; }