2020-06-22 16:45:57 +02:00
|
|
|
#pragma once
|
2020-09-18 00:23:03 +02:00
|
|
|
#include <dsp/block.h>
|
2020-11-02 03:57:44 +01:00
|
|
|
#include <dsp/window.h>
|
|
|
|
#include <dsp/resampling.h>
|
|
|
|
#include <dsp/processing.h>
|
|
|
|
#include <algorithm>
|
2020-06-22 16:45:57 +02:00
|
|
|
|
|
|
|
namespace dsp {
|
|
|
|
class VFO {
|
|
|
|
public:
|
2020-11-02 03:57:44 +01:00
|
|
|
VFO() {}
|
2020-06-22 16:45:57 +02:00
|
|
|
|
2021-07-12 05:03:51 +02:00
|
|
|
~VFO() {
|
|
|
|
if (!_init) { return; }
|
|
|
|
stop();
|
|
|
|
_init = false;
|
|
|
|
}
|
2020-11-02 03:57:44 +01:00
|
|
|
|
|
|
|
VFO(stream<complex_t>* in, float offset, float inSampleRate, float outSampleRate, float bandWidth) {
|
|
|
|
init(in, offset, inSampleRate, outSampleRate, bandWidth);
|
|
|
|
};
|
2020-06-22 16:45:57 +02:00
|
|
|
|
2020-11-02 03:57:44 +01:00
|
|
|
void init(stream<complex_t>* in, float offset, float inSampleRate, float outSampleRate, float bandWidth) {
|
|
|
|
_in = in;
|
|
|
|
_offset = offset;
|
|
|
|
_inSampleRate = inSampleRate;
|
|
|
|
_outSampleRate = outSampleRate;
|
2020-06-22 16:45:57 +02:00
|
|
|
_bandWidth = bandWidth;
|
|
|
|
|
2020-11-02 03:57:44 +01:00
|
|
|
float realCutoff = std::min<float>(_bandWidth, std::min<float>(_inSampleRate, _outSampleRate)) / 2.0f;
|
|
|
|
|
|
|
|
xlator.init(_in, _inSampleRate, -_offset);
|
|
|
|
win.init(realCutoff, realCutoff, inSampleRate);
|
|
|
|
resamp.init(&xlator.out, &win, _inSampleRate, _outSampleRate);
|
|
|
|
|
|
|
|
win.setSampleRate(_inSampleRate * resamp.getInterpolation());
|
|
|
|
resamp.updateWindow(&win);
|
|
|
|
|
|
|
|
out = &resamp.out;
|
2021-07-12 05:03:51 +02:00
|
|
|
|
|
|
|
_init = true;
|
2020-06-22 16:45:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void start() {
|
2021-07-12 05:03:51 +02:00
|
|
|
assert(_init);
|
2020-11-02 03:57:44 +01:00
|
|
|
if (running) { return; }
|
|
|
|
xlator.start();
|
2020-07-19 15:59:44 +02:00
|
|
|
resamp.start();
|
2020-06-22 16:45:57 +02:00
|
|
|
}
|
|
|
|
|
2020-11-02 03:57:44 +01:00
|
|
|
void stop() {
|
2021-07-12 05:03:51 +02:00
|
|
|
assert(_init);
|
2020-11-02 03:57:44 +01:00
|
|
|
if (!running) { return; }
|
|
|
|
xlator.stop();
|
|
|
|
resamp.stop();
|
2020-06-22 16:45:57 +02:00
|
|
|
}
|
|
|
|
|
2020-11-02 03:57:44 +01:00
|
|
|
void setInSampleRate(float inSampleRate) {
|
2021-07-12 05:03:51 +02:00
|
|
|
assert(_init);
|
2020-11-02 03:57:44 +01:00
|
|
|
_inSampleRate = inSampleRate;
|
|
|
|
if (running) { xlator.stop(); resamp.stop(); }
|
|
|
|
xlator.setSampleRate(_inSampleRate);
|
|
|
|
resamp.setInSampleRate(_inSampleRate);
|
|
|
|
float realCutoff = std::min<float>(_bandWidth, std::min<float>(_inSampleRate, _outSampleRate)) / 2.0f;
|
|
|
|
win.setSampleRate(_inSampleRate * resamp.getInterpolation());
|
|
|
|
win.setCutoff(realCutoff);
|
|
|
|
win.setTransWidth(realCutoff);
|
|
|
|
resamp.updateWindow(&win);
|
|
|
|
if (running) { xlator.start(); resamp.start(); }
|
2020-06-22 16:45:57 +02:00
|
|
|
}
|
|
|
|
|
2020-11-02 03:57:44 +01:00
|
|
|
void setOutSampleRate(float outSampleRate) {
|
2021-07-12 05:03:51 +02:00
|
|
|
assert(_init);
|
2020-11-02 03:57:44 +01:00
|
|
|
_outSampleRate = outSampleRate;
|
|
|
|
if (running) { resamp.stop(); }
|
|
|
|
resamp.setOutSampleRate(_outSampleRate);
|
|
|
|
float realCutoff = std::min<float>(_bandWidth, std::min<float>(_inSampleRate, _outSampleRate)) / 2.0f;
|
|
|
|
win.setSampleRate(_inSampleRate * resamp.getInterpolation());
|
|
|
|
win.setCutoff(realCutoff);
|
|
|
|
win.setTransWidth(realCutoff);
|
|
|
|
resamp.updateWindow(&win);
|
|
|
|
if (running) { resamp.start(); }
|
2020-06-22 16:45:57 +02:00
|
|
|
}
|
|
|
|
|
2020-11-02 03:57:44 +01:00
|
|
|
void setOutSampleRate(float outSampleRate, float bandWidth) {
|
2021-07-12 05:03:51 +02:00
|
|
|
assert(_init);
|
2020-11-02 03:57:44 +01:00
|
|
|
_outSampleRate = outSampleRate;
|
2020-07-19 15:59:44 +02:00
|
|
|
_bandWidth = bandWidth;
|
2020-11-02 03:57:44 +01:00
|
|
|
if (running) { resamp.stop(); }
|
|
|
|
resamp.setOutSampleRate(_outSampleRate);
|
|
|
|
float realCutoff = std::min<float>(_bandWidth, std::min<float>(_inSampleRate, _outSampleRate)) / 2.0f;
|
|
|
|
win.setSampleRate(_inSampleRate * resamp.getInterpolation());
|
|
|
|
win.setCutoff(realCutoff);
|
|
|
|
win.setTransWidth(realCutoff);
|
|
|
|
resamp.updateWindow(&win);
|
|
|
|
if (running) { resamp.start(); }
|
2020-06-22 16:45:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void setOffset(float offset) {
|
2021-07-12 05:03:51 +02:00
|
|
|
assert(_init);
|
2020-11-02 03:57:44 +01:00
|
|
|
_offset = offset;
|
|
|
|
xlator.setFrequency(-_offset);
|
2020-06-22 16:45:57 +02:00
|
|
|
}
|
|
|
|
|
2020-11-02 03:57:44 +01:00
|
|
|
void setBandwidth(float bandWidth) {
|
2021-07-12 05:03:51 +02:00
|
|
|
assert(_init);
|
2020-11-02 03:57:44 +01:00
|
|
|
_bandWidth = bandWidth;
|
|
|
|
float realCutoff = std::min<float>(_bandWidth, std::min<float>(_inSampleRate, _outSampleRate)) / 2.0f;
|
|
|
|
win.setCutoff(realCutoff);
|
|
|
|
win.setTransWidth(realCutoff);
|
|
|
|
resamp.updateWindow(&win);
|
2020-07-19 15:59:44 +02:00
|
|
|
}
|
|
|
|
|
2020-11-02 03:57:44 +01:00
|
|
|
stream<complex_t>* out;
|
2020-06-22 16:45:57 +02:00
|
|
|
|
|
|
|
private:
|
2021-07-12 05:03:51 +02:00
|
|
|
bool _init = false;
|
2020-11-02 03:57:44 +01:00
|
|
|
bool running = false;
|
|
|
|
float _offset, _inSampleRate, _outSampleRate, _bandWidth;
|
|
|
|
filter_window::BlackmanWindow win;
|
|
|
|
stream<complex_t>* _in;
|
2020-12-06 16:13:47 +01:00
|
|
|
FrequencyXlator<complex_t> xlator;
|
2020-11-02 03:57:44 +01:00
|
|
|
PolyphaseResampler<complex_t> resamp;
|
|
|
|
|
2020-06-22 16:45:57 +02:00
|
|
|
};
|
2020-11-02 03:57:44 +01:00
|
|
|
}
|