/////////////////////////////////////////////////////////////////////////////////// // Copyright (C) 2020 Edouard Griffiths, F4EXB // // Copyright (C) 2020 Kacper Michajłow // // Copyright (C) 2021, 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 . // /////////////////////////////////////////////////////////////////////////////////// #pragma once #include #include #include "dsp/dsptypes.h" #include "dsp/misc.h" #include "export.h" namespace FirFilterGenerators { SDRBASE_API void generateLowPassFilter(int nTaps, double sampleRate, double cutoff, std::vector &taps); }; template class FirFilter { public: Type filter(Type sample) { Type acc = 0; unsigned int n_samples = m_samples.size(); unsigned int n_taps = m_taps.size() - 1; unsigned int a = m_ptr; unsigned int b = a == n_samples - 1 ? 0 : a + 1; m_samples[m_ptr] = sample; for (unsigned int i = 0; i < n_taps; ++i) { acc += (m_samples[a] + m_samples[b]) * m_taps[i]; a = (a == 0) ? n_samples - 1 : a - 1; b = (b == n_samples - 1) ? 0 : b + 1; } acc += m_samples[a] * m_taps[n_taps]; m_ptr = (m_ptr == n_samples - 1) ? 0 : m_ptr + 1; return acc; } // Print taps as a Matlab vector void printTaps(const char *name) { printf("%s = [", name); for (int i = 0; i <= m_taps.size() - 1; ++i) { printf("%g ", m_taps[i]); } for (int i = m_taps.size() - 2; i >= 0; --i) { printf("%g ", m_taps[i]); } printf("];\n"); } protected: void init(int nTaps) { m_ptr = 0; m_samples.resize(nTaps); for (int i = 0; i < nTaps; i++) { m_samples[i] = 0; } } protected: std::vector m_taps; std::vector m_samples; size_t m_ptr; }; template struct Lowpass : public FirFilter { public: void create(int nTaps, double sampleRate, double cutoff) { this->init(nTaps); FirFilterGenerators::generateLowPassFilter(nTaps, sampleRate, cutoff, this->m_taps); } }; template struct Bandpass : public FirFilter { void create(int nTaps, double sampleRate, double lowCutoff, double highCutoff) { this->init(nTaps); FirFilterGenerators::generateLowPassFilter(nTaps, sampleRate, highCutoff, this->m_taps); std::vector highPass; FirFilterGenerators::generateLowPassFilter(nTaps, sampleRate, lowCutoff, highPass); for (size_t i = 0; i < highPass.size(); ++i) { highPass[i] = -highPass[i]; } highPass[highPass.size() - 1] += 1; for (size_t i = 0; i < this->m_taps.size(); ++i) { this->m_taps[i] = -(this->m_taps[i] + highPass[i]); } this->m_taps[this->m_taps.size() - 1] += 1; } }; template struct Highpass : public FirFilter { void create(int nTaps, double sampleRate, double cutoff) { this->init(nTaps); FirFilterGenerators::generateLowPassFilter(nTaps, sampleRate, cutoff, this->m_taps); for (size_t i = 0; i < this->m_taps.size(); ++i) { this->m_taps[i] = -this->m_taps[i]; } this->m_taps[this->m_taps.size() - 1] += 1; } };