mirror of
https://github.com/f4exb/sdrangel.git
synced 2024-11-29 19:28:47 -05:00
580 lines
13 KiB
C++
580 lines
13 KiB
C++
// Copyright (C) 2018-2019, 2021, 2023 Edouard Griffiths, F4EXB <f4exb06@gmail.com> //
|
|
//
|
|
// This file is part of LeanSDR Copyright (C) 2016-2018 <pabr@pabr.org>.
|
|
//
|
|
// 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, either 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 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 LEANSDR_DSP_H
|
|
#define LEANSDR_DSP_H
|
|
|
|
#include "leansdr/framework.h"
|
|
#include "leansdr/math.h"
|
|
#include <math.h>
|
|
|
|
namespace leansdr
|
|
{
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// DSP blocks
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
// [cconverter] converts std::complex streams between numric types,
|
|
// with optional ofsetting and rational scaling.
|
|
template <typename Tin, int Zin, typename Tout, int Zout, int Gn, int Gd>
|
|
struct cconverter : runnable
|
|
{
|
|
cconverter(
|
|
scheduler *sch,
|
|
pipebuf<std::complex<Tin>> &_in,
|
|
pipebuf<std::complex<Tout>> &_out
|
|
) :
|
|
runnable(sch, "cconverter"),
|
|
in(_in),
|
|
out(_out)
|
|
{
|
|
}
|
|
|
|
void run()
|
|
{
|
|
unsigned long count = min(in.readable(), out.writable());
|
|
std::complex<Tin> *pin = in.rd(), *pend = pin + count;
|
|
std::complex<Tout> *pout = out.wr();
|
|
|
|
for (; pin < pend; ++pin, ++pout)
|
|
{
|
|
pout->re = Zout + (pin->re - (Tin)Zin) * Gn / Gd;
|
|
pout->im = Zout + (pin->im - (Tin)Zin) * Gn / Gd;
|
|
}
|
|
|
|
in.read(count);
|
|
out.written(count);
|
|
}
|
|
|
|
private:
|
|
pipereader<std::complex<Tin>> in;
|
|
pipewriter<std::complex<Tout>> out;
|
|
};
|
|
|
|
template <typename T>
|
|
struct cfft_engine
|
|
{
|
|
cfft_engine(int _n) :
|
|
bitrev(nullptr),
|
|
omega(nullptr),
|
|
omega_rev(nullptr)
|
|
{
|
|
init(_n);
|
|
}
|
|
|
|
~cfft_engine() {
|
|
release();
|
|
}
|
|
|
|
int size() {
|
|
return n;
|
|
}
|
|
|
|
void init(int _n)
|
|
{
|
|
release();
|
|
n = _n;
|
|
invsqrtn = 1.0 / sqrt(n);
|
|
|
|
// Compute log2(n)
|
|
logn = 0;
|
|
for (int t = n; t > 1; t >>= 1) {
|
|
++logn;
|
|
}
|
|
|
|
// Bit reversal
|
|
bitrev = new int[n];
|
|
|
|
for (int i = 0; i < n; ++i)
|
|
{
|
|
bitrev[i] = 0;
|
|
|
|
for (int b = 0; b < logn; ++b) {
|
|
bitrev[i] = (bitrev[i] << 1) | ((i >> b) & 1);
|
|
}
|
|
}
|
|
|
|
// Float constants
|
|
omega = new std::complex<T>[n];
|
|
omega_rev = new std::complex<T>[n];
|
|
|
|
for (int i = 0; i < n; ++i)
|
|
{
|
|
float a = 2.0 * M_PI * i / n;
|
|
omega_rev[i].real(cosf(a));
|
|
omega[i].real(cosf(a));
|
|
omega_rev[i].imag(sinf(a));
|
|
omega[i].imag(sinf(a));
|
|
// omega_rev[i].re = (omega[i].re = cosf(a));
|
|
// omega_rev[i].im = -(omega[i].im = sinf(a));
|
|
}
|
|
}
|
|
|
|
void inplace(std::complex<T> *data, bool reverse = false)
|
|
{
|
|
// Bit-reversal permutation
|
|
for (int i = 0; i < n; ++i)
|
|
{
|
|
int r = bitrev[i];
|
|
|
|
if ((r < 0) || (r >= n)) // corruption: init again and exit
|
|
{
|
|
fprintf(stderr, "cfft_engine::inplace: corruption detected\n");
|
|
init(n);
|
|
return;
|
|
}
|
|
|
|
if (r < i)
|
|
{
|
|
std::complex<T> tmp = data[i];
|
|
data[i] = data[r];
|
|
data[r] = tmp;
|
|
}
|
|
}
|
|
|
|
std::complex<T> *om = reverse ? omega_rev : omega;
|
|
// Danielson-Lanczos
|
|
for (int i = 0; i < logn; ++i)
|
|
{
|
|
int hbs = 1 << i;
|
|
int dom = 1 << (logn - 1 - i);
|
|
|
|
for (int j = 0; j < dom; ++j)
|
|
{
|
|
int p = j * hbs * 2, q = p + hbs;
|
|
|
|
for (int k = 0; k < hbs; ++k)
|
|
{
|
|
std::complex<T> &w = om[k * dom];
|
|
std::complex<T> &dqk = data[q + k];
|
|
std::complex<T> x(w.real() * dqk.real() - w.imag() * dqk.imag(),
|
|
w.real() * dqk.imag() + w.imag() * dqk.real());
|
|
data[q + k].real(data[p + k].real() - x.real());
|
|
data[q + k].imag(data[p + k].imag() - x.imag());
|
|
data[p + k].real(data[p + k].real() + x.real());
|
|
data[p + k].imag(data[p + k].imag() + x.imag());
|
|
}
|
|
}
|
|
}
|
|
|
|
if (reverse)
|
|
{
|
|
float invn = 1.0 / n;
|
|
|
|
for (int i = 0; i < n; ++i) {
|
|
data[i] *= invn;
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
void release()
|
|
{
|
|
if (bitrev) {
|
|
delete[] bitrev;
|
|
}
|
|
if (omega) {
|
|
delete[] omega;
|
|
}
|
|
if (omega_rev) {
|
|
delete[] omega_rev;
|
|
}
|
|
}
|
|
|
|
int *bitrev;
|
|
std::complex<T> *omega;
|
|
std::complex<T> *omega_rev;
|
|
int n;
|
|
float invsqrtn;
|
|
int logn;
|
|
};
|
|
|
|
template <typename T>
|
|
struct adder : runnable
|
|
{
|
|
adder(
|
|
scheduler *sch,
|
|
pipebuf<T> &_in1,
|
|
pipebuf<T> &_in2,
|
|
pipebuf<T> &_out
|
|
) :
|
|
runnable(sch, "adder"),
|
|
in1(_in1),
|
|
in2(_in2),
|
|
out(_out)
|
|
{
|
|
}
|
|
|
|
void run()
|
|
{
|
|
int n = out.writable();
|
|
|
|
if (in1.readable() < n) {
|
|
n = in1.readable();
|
|
}
|
|
|
|
if (in2.readable() < n) {
|
|
n = in2.readable();
|
|
}
|
|
|
|
T *pin1 = in1.rd(), *pin2 = in2.rd(), *pout = out.wr(), *pend = pout + n;
|
|
|
|
while (pout < pend) {
|
|
*pout++ = *pin1++ + *pin2++;
|
|
}
|
|
|
|
in1.read(n);
|
|
in2.read(n);
|
|
out.written(n);
|
|
}
|
|
|
|
private:
|
|
pipereader<T> in1, in2;
|
|
pipewriter<T> out;
|
|
};
|
|
|
|
template <typename Tscale, typename Tin, typename Tout>
|
|
struct scaler : runnable
|
|
{
|
|
Tscale scale;
|
|
|
|
scaler(
|
|
scheduler *sch,
|
|
Tscale _scale,
|
|
pipebuf<Tin> &_in,
|
|
pipebuf<Tout> &_out
|
|
) :
|
|
runnable(sch, "scaler"),
|
|
scale(_scale),
|
|
in(_in),
|
|
out(_out)
|
|
{
|
|
}
|
|
|
|
void run()
|
|
{
|
|
unsigned long count = min(in.readable(), out.writable());
|
|
Tin *pin = in.rd(), *pend = pin + count;
|
|
Tout *pout = out.wr();
|
|
|
|
for (; pin < pend; ++pin, ++pout) {
|
|
*pout = *pin * scale;
|
|
}
|
|
|
|
in.read(count);
|
|
out.written(count);
|
|
}
|
|
|
|
private:
|
|
pipereader<Tin> in;
|
|
pipewriter<Tout> out;
|
|
};
|
|
|
|
// [awgb_c] generates std::complex white gaussian noise.
|
|
|
|
template <typename T>
|
|
struct wgn_c : runnable
|
|
{
|
|
wgn_c(
|
|
scheduler *sch,
|
|
pipebuf<std::complex<T>>
|
|
&_out
|
|
) :
|
|
runnable(sch, "awgn"),
|
|
stddev(1.0),
|
|
out(_out)
|
|
{
|
|
}
|
|
|
|
void run()
|
|
{
|
|
int n = out.writable();
|
|
std::complex<T> *pout = out.wr(), *pend = pout + n;
|
|
|
|
while (pout < pend)
|
|
{
|
|
// TAOCP
|
|
float x, y, r2;
|
|
|
|
do
|
|
{
|
|
x = 2 * rand_compat() - 1;
|
|
y = 2 * rand_compat() - 1;
|
|
r2 = x * x + y * y;
|
|
} while (r2 == 0 || r2 >= 1);
|
|
|
|
float k = sqrtf(-logf(r2) / r2) * stddev;
|
|
pout->re = k * x;
|
|
pout->im = k * y;
|
|
++pout;
|
|
}
|
|
|
|
out.written(n);
|
|
}
|
|
float stddev;
|
|
|
|
private:
|
|
pipewriter<std::complex<T>> out;
|
|
};
|
|
|
|
template <typename T>
|
|
struct naive_lowpass : runnable
|
|
{
|
|
naive_lowpass(
|
|
scheduler *sch,
|
|
pipebuf<T> &_in,
|
|
pipebuf<T> &_out,
|
|
int _w
|
|
) :
|
|
runnable(sch, "lowpass"),
|
|
in(_in),
|
|
out(_out),
|
|
w(_w)
|
|
{
|
|
}
|
|
|
|
void run()
|
|
{
|
|
if (in.readable() < w) {
|
|
return;
|
|
}
|
|
|
|
unsigned long count = min(in.readable() - w, out.writable());
|
|
T *pin = in.rd(), *pend = pin + count;
|
|
T *pout = out.wr();
|
|
float k = 1.0 / w;
|
|
|
|
for (; pin < pend; ++pin, ++pout)
|
|
{
|
|
T x = 0.0;
|
|
|
|
for (int i = 0; i < w; ++i) {
|
|
x = x + pin[i];
|
|
}
|
|
|
|
*pout = x * k;
|
|
}
|
|
|
|
in.read(count);
|
|
out.written(count);
|
|
}
|
|
|
|
private:
|
|
pipereader<T> in;
|
|
pipewriter<T> out;
|
|
int w;
|
|
};
|
|
|
|
template <typename T, typename Tc>
|
|
struct fir_filter : runnable
|
|
{
|
|
fir_filter(
|
|
scheduler *sch,
|
|
int _ncoeffs,
|
|
Tc *_coeffs,
|
|
pipebuf<T> &_in,
|
|
pipebuf<T> &_out,
|
|
unsigned int _decim = 1
|
|
) :
|
|
runnable(sch, "fir_filter"),
|
|
ncoeffs(_ncoeffs),
|
|
coeffs(_coeffs),
|
|
in(_in),
|
|
out(_out),
|
|
decim(_decim),
|
|
freq_tap(nullptr),
|
|
tap_multiplier(1),
|
|
freq_tol(0.1)
|
|
{
|
|
shifted_coeffs = new T[ncoeffs];
|
|
set_freq(0);
|
|
}
|
|
|
|
~fir_filter() {
|
|
delete[] shifted_coeffs;
|
|
}
|
|
|
|
void run()
|
|
{
|
|
if (in.readable() < ncoeffs)
|
|
return;
|
|
|
|
if (freq_tap)
|
|
{
|
|
float new_freq = *freq_tap * tap_multiplier;
|
|
if (fabs(current_freq - new_freq) > freq_tol)
|
|
{
|
|
if (sch->verbose)
|
|
fprintf(stderr, "Shifting filter %f -> %f\n",
|
|
current_freq, new_freq);
|
|
set_freq(new_freq);
|
|
}
|
|
}
|
|
|
|
long count = min((in.readable() - ncoeffs) / decim,
|
|
out.writable());
|
|
T *pin = in.rd() + ncoeffs, *pend = pin + count * decim, *pout = out.wr();
|
|
// TBD use coeffs when current_freq=0 (fewer mults if float)
|
|
for (; pin < pend; pin += decim, ++pout)
|
|
{
|
|
T *pc = shifted_coeffs;
|
|
T *pi = pin;
|
|
T x = 0;
|
|
for (int i = ncoeffs; i--; ++pc, --pi)
|
|
x = x + (*pc) * (*pi);
|
|
*pout = x;
|
|
}
|
|
in.read(count * decim);
|
|
out.written(count);
|
|
}
|
|
|
|
public:
|
|
float *freq_tap;
|
|
float tap_multiplier;
|
|
float freq_tol;
|
|
|
|
private:
|
|
int ncoeffs;
|
|
Tc *coeffs;
|
|
pipereader<T> in;
|
|
pipewriter<T> out;
|
|
int decim;
|
|
T *shifted_coeffs;
|
|
float current_freq;
|
|
|
|
void set_freq(float f)
|
|
{
|
|
for (int i = 0; i < ncoeffs; ++i)
|
|
{
|
|
float a = 2 * M_PI * f * (i - ncoeffs / 2.0);
|
|
float c = cosf(a), s = sinf(a);
|
|
// TBD Support T=std::complex
|
|
shifted_coeffs[i].real(coeffs[i] * c);
|
|
shifted_coeffs[i].imag(coeffs[i] * s);
|
|
}
|
|
current_freq = f;
|
|
}
|
|
}; // fir_filter
|
|
|
|
// FIR FILTER WITH INTERPOLATION AND DECIMATION
|
|
|
|
template <typename T, typename Tc>
|
|
struct fir_resampler : runnable
|
|
{
|
|
fir_resampler(
|
|
scheduler *sch,
|
|
int _ncoeffs,
|
|
Tc *_coeffs,
|
|
pipebuf<T> &_in,
|
|
pipebuf<T> &_out,
|
|
int _interp = 1,
|
|
int _decim = 1
|
|
) :
|
|
runnable(sch, "fir_resampler"),
|
|
ncoeffs(_ncoeffs),
|
|
coeffs(_coeffs),
|
|
interp(_interp),
|
|
decim(_decim),
|
|
in(_in),
|
|
out(_out, interp),
|
|
freq_tap(nullptr),
|
|
tap_multiplier(1),
|
|
freq_tol(0.1)
|
|
{
|
|
if (decim != 1)
|
|
fail("fir_resampler: decim not implemented"); // TBD
|
|
shifted_coeffs = new T[ncoeffs];
|
|
set_freq(0);
|
|
}
|
|
|
|
~fir_resampler() {
|
|
delete[] shifted_coeffs;
|
|
}
|
|
|
|
void run()
|
|
{
|
|
if (in.readable() < ncoeffs)
|
|
return;
|
|
|
|
if (freq_tap)
|
|
{
|
|
float new_freq = *freq_tap * tap_multiplier;
|
|
if (fabs(current_freq - new_freq) > freq_tol)
|
|
{
|
|
if (sch->verbose)
|
|
fprintf(stderr, "Shifting filter %f -> %f\n",
|
|
current_freq, new_freq);
|
|
set_freq(new_freq);
|
|
}
|
|
}
|
|
|
|
if (in.readable() * interp < ncoeffs)
|
|
return;
|
|
unsigned long count = min((in.readable() * interp - ncoeffs) / interp,
|
|
out.writable() / interp);
|
|
int latency = (ncoeffs + interp) / interp;
|
|
T *pin = in.rd() + latency, *pend = pin + count, *pout = out.wr();
|
|
// TBD use coeffs when current_freq=0 (fewer mults if float)
|
|
for (; pin < pend; ++pin)
|
|
{
|
|
for (int i = 0; i < interp; ++i, ++pout)
|
|
{
|
|
T *pi = pin;
|
|
T *pc = shifted_coeffs + i, *pcend = shifted_coeffs + ncoeffs;
|
|
T x = 0;
|
|
for (; pc < pcend; pc += interp, --pi)
|
|
x = x + (*pc) * (*pi);
|
|
*pout = x;
|
|
}
|
|
}
|
|
in.read(count);
|
|
out.written(count * interp);
|
|
}
|
|
|
|
public:
|
|
float *freq_tap;
|
|
float tap_multiplier;
|
|
float freq_tol;
|
|
|
|
private:
|
|
unsigned int ncoeffs;
|
|
Tc *coeffs;
|
|
int interp, decim;
|
|
pipereader<T> in;
|
|
pipewriter<T> out;
|
|
T *shifted_coeffs;
|
|
float current_freq;
|
|
|
|
void set_freq(float f)
|
|
{
|
|
for (int i = 0; i < ncoeffs; ++i)
|
|
{
|
|
float a = 2 * M_PI * f * i;
|
|
float c = cosf(a), s = sinf(a);
|
|
// TBD Support T=std::complex
|
|
shifted_coeffs[i].re = coeffs[i] * c;
|
|
shifted_coeffs[i].im = coeffs[i] * s;
|
|
}
|
|
current_freq = f;
|
|
}
|
|
}; // fir_resampler
|
|
|
|
} // namespace leansdr
|
|
|
|
#endif // LEANSDR_DSP_H
|