2020-06-10 04:13:56 +02:00
|
|
|
#pragma once
|
|
|
|
#include <thread>
|
2020-06-22 16:45:57 +02:00
|
|
|
#include <dsp/stream.h>
|
|
|
|
#include <dsp/types.h>
|
|
|
|
|
|
|
|
/*
|
|
|
|
TODO:
|
|
|
|
- Add a sample rate ajustment function to all demodulators
|
|
|
|
*/
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2020-06-15 15:53:45 +02:00
|
|
|
#define FAST_ATAN2_COEF1 3.1415926535f / 4.0f
|
|
|
|
#define FAST_ATAN2_COEF2 3.0f * FAST_ATAN2_COEF1
|
|
|
|
|
2020-06-22 16:45:57 +02:00
|
|
|
inline float fast_arctan2(float y, float x) {
|
2020-06-15 15:53:45 +02:00
|
|
|
float abs_y = fabs(y)+1e-10;
|
|
|
|
float r, angle;
|
|
|
|
if (x>=0)
|
|
|
|
{
|
|
|
|
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) {
|
|
|
|
return -angle;
|
|
|
|
}
|
|
|
|
return angle;
|
|
|
|
}
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2020-06-22 16:45:57 +02:00
|
|
|
namespace dsp {
|
2020-06-10 04:13:56 +02:00
|
|
|
class FMDemodulator {
|
|
|
|
public:
|
2020-06-10 18:52:07 +02:00
|
|
|
FMDemodulator() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-06-22 16:45:57 +02:00
|
|
|
FMDemodulator(stream<complex_t>* in, float deviation, long sampleRate, int blockSize) : output(blockSize * 2) {
|
2020-06-15 15:53:45 +02:00
|
|
|
running = false;
|
2020-06-10 04:13:56 +02:00
|
|
|
_input = in;
|
2020-06-22 16:45:57 +02:00
|
|
|
_blockSize = blockSize;
|
2020-06-10 04:13:56 +02:00
|
|
|
_phase = 0.0f;
|
|
|
|
_phasorSpeed = (2 * 3.1415926535) / (sampleRate / deviation);
|
|
|
|
}
|
|
|
|
|
2020-06-22 16:45:57 +02:00
|
|
|
void init(stream<complex_t>* in, float deviation, long sampleRate, int blockSize) {
|
|
|
|
output.init(blockSize * 2);
|
2020-06-15 15:53:45 +02:00
|
|
|
running = false;
|
2020-06-10 18:52:07 +02:00
|
|
|
_input = in;
|
2020-06-22 16:45:57 +02:00
|
|
|
_blockSize = blockSize;
|
2020-06-10 18:52:07 +02:00
|
|
|
_phase = 0.0f;
|
|
|
|
_phasorSpeed = (2 * 3.1415926535) / (sampleRate / deviation);
|
|
|
|
}
|
|
|
|
|
2020-06-10 04:13:56 +02:00
|
|
|
void start() {
|
2020-06-15 15:53:45 +02:00
|
|
|
if (running) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
running = true;
|
2020-06-10 04:13:56 +02:00
|
|
|
_workerThread = std::thread(_worker, this);
|
|
|
|
}
|
|
|
|
|
2020-06-15 15:53:45 +02:00
|
|
|
void stop() {
|
|
|
|
if (!running) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
_input->stopReader();
|
|
|
|
output.stopWriter();
|
|
|
|
_workerThread.join();
|
|
|
|
running = false;
|
|
|
|
_input->clearReadStop();
|
|
|
|
output.clearWriteStop();
|
|
|
|
}
|
|
|
|
|
2020-06-22 16:45:57 +02:00
|
|
|
void setBlockSize(int blockSize) {
|
|
|
|
if (running) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
_blockSize = blockSize;
|
|
|
|
output.setMaxLatency(_blockSize * 2);
|
|
|
|
}
|
|
|
|
|
2020-06-10 04:13:56 +02:00
|
|
|
stream<float> output;
|
|
|
|
|
|
|
|
private:
|
|
|
|
static void _worker(FMDemodulator* _this) {
|
2020-06-22 16:45:57 +02:00
|
|
|
complex_t* inBuf = new complex_t[_this->_blockSize];
|
|
|
|
float* outBuf = new float[_this->_blockSize];
|
2020-06-10 04:13:56 +02:00
|
|
|
float diff = 0;
|
|
|
|
float currentPhase = 0;
|
|
|
|
while (true) {
|
2020-06-22 16:45:57 +02:00
|
|
|
if (_this->_input->read(inBuf, _this->_blockSize) < 0) { return; };
|
|
|
|
for (int i = 0; i < _this->_blockSize; i++) {
|
2020-06-15 15:53:45 +02:00
|
|
|
currentPhase = fast_arctan2(inBuf[i].i, inBuf[i].q);
|
2020-06-10 04:13:56 +02:00
|
|
|
diff = currentPhase - _this->_phase;
|
2020-06-15 15:53:45 +02:00
|
|
|
if (diff > 3.1415926535f) { diff -= 2 * 3.1415926535f; }
|
|
|
|
else if (diff <= -3.1415926535f) { diff += 2 * 3.1415926535f; }
|
2020-06-10 04:13:56 +02:00
|
|
|
outBuf[i] = diff / _this->_phasorSpeed;
|
|
|
|
_this->_phase = currentPhase;
|
|
|
|
}
|
2020-06-22 16:45:57 +02:00
|
|
|
if (_this->output.write(outBuf, _this->_blockSize) < 0) { return; };
|
2020-06-10 04:13:56 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
stream<complex_t>* _input;
|
2020-06-15 15:53:45 +02:00
|
|
|
bool running;
|
2020-06-22 16:45:57 +02:00
|
|
|
int _blockSize;
|
2020-06-10 04:13:56 +02:00
|
|
|
float _phase;
|
|
|
|
float _phasorSpeed;
|
|
|
|
std::thread _workerThread;
|
|
|
|
};
|
2020-06-15 15:53:45 +02:00
|
|
|
|
|
|
|
|
|
|
|
class AMDemodulator {
|
|
|
|
public:
|
|
|
|
AMDemodulator() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2020-06-22 16:45:57 +02:00
|
|
|
AMDemodulator(stream<complex_t>* in, int blockSize) : output(blockSize * 2) {
|
2020-06-15 15:53:45 +02:00
|
|
|
running = false;
|
|
|
|
_input = in;
|
2020-06-22 16:45:57 +02:00
|
|
|
_blockSize = blockSize;
|
2020-06-15 15:53:45 +02:00
|
|
|
}
|
|
|
|
|
2020-06-22 16:45:57 +02:00
|
|
|
void init(stream<complex_t>* in, int blockSize) {
|
|
|
|
output.init(blockSize * 2);
|
2020-06-15 15:53:45 +02:00
|
|
|
running = false;
|
|
|
|
_input = in;
|
2020-06-22 16:45:57 +02:00
|
|
|
_blockSize = blockSize;
|
2020-06-15 15:53:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void start() {
|
|
|
|
if (running) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
running = true;
|
|
|
|
_workerThread = std::thread(_worker, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void stop() {
|
|
|
|
if (!running) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
_input->stopReader();
|
|
|
|
output.stopWriter();
|
|
|
|
_workerThread.join();
|
|
|
|
running = false;
|
|
|
|
_input->clearReadStop();
|
|
|
|
output.clearWriteStop();
|
|
|
|
}
|
|
|
|
|
2020-06-22 16:45:57 +02:00
|
|
|
void setBlockSize(int blockSize) {
|
|
|
|
if (running) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
_blockSize = blockSize;
|
|
|
|
output.setMaxLatency(_blockSize * 2);
|
|
|
|
}
|
|
|
|
|
2020-06-15 15:53:45 +02:00
|
|
|
stream<float> output;
|
|
|
|
|
|
|
|
private:
|
|
|
|
static void _worker(AMDemodulator* _this) {
|
2020-06-22 16:45:57 +02:00
|
|
|
complex_t* inBuf = new complex_t[_this->_blockSize];
|
|
|
|
float* outBuf = new float[_this->_blockSize];
|
2020-06-15 15:53:45 +02:00
|
|
|
float min, max, amp;
|
|
|
|
while (true) {
|
2020-06-22 16:45:57 +02:00
|
|
|
if (_this->_input->read(inBuf, _this->_blockSize) < 0) { break; };
|
2020-06-15 15:53:45 +02:00
|
|
|
min = INFINITY;
|
|
|
|
max = 0.0f;
|
2020-06-22 16:45:57 +02:00
|
|
|
for (int i = 0; i < _this->_blockSize; i++) {
|
2020-06-15 15:53:45 +02:00
|
|
|
outBuf[i] = sqrt((inBuf[i].i*inBuf[i].i) + (inBuf[i].q*inBuf[i].q));
|
|
|
|
if (outBuf[i] < min) {
|
|
|
|
min = outBuf[i];
|
|
|
|
}
|
|
|
|
if (outBuf[i] > max) {
|
|
|
|
max = outBuf[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
amp = (max - min);
|
2020-06-22 16:45:57 +02:00
|
|
|
for (int i = 0; i < _this->_blockSize; i++) {
|
2020-06-15 15:53:45 +02:00
|
|
|
outBuf[i] = (outBuf[i] - min) / (max - min);
|
|
|
|
}
|
2020-06-22 16:45:57 +02:00
|
|
|
if (_this->output.write(outBuf, _this->_blockSize) < 0) { break; };
|
2020-06-15 15:53:45 +02:00
|
|
|
}
|
2020-06-22 16:45:57 +02:00
|
|
|
delete[] inBuf;
|
|
|
|
delete[] outBuf;
|
2020-06-15 15:53:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
stream<complex_t>* _input;
|
|
|
|
bool running;
|
2020-06-22 16:45:57 +02:00
|
|
|
int _blockSize;
|
2020-06-15 15:53:45 +02:00
|
|
|
std::thread _workerThread;
|
|
|
|
};
|
2020-06-10 04:13:56 +02:00
|
|
|
};
|