mirror of
https://github.com/AlexandreRouma/SDRPlusPlus.git
synced 2025-01-11 18:57:11 +01:00
Added basic RDS support, no error correction yet
This commit is contained in:
parent
46f17019a7
commit
edf22ccfe8
189
core/src/dsp/clock_recovery/fd.h
Normal file
189
core/src/dsp/clock_recovery/fd.h
Normal file
@ -0,0 +1,189 @@
|
|||||||
|
#pragma once
|
||||||
|
#include "../processor.h"
|
||||||
|
#include "../loop/phase_control_loop.h"
|
||||||
|
#include "../taps/windowed_sinc.h"
|
||||||
|
#include "../multirate/polyphase_bank.h"
|
||||||
|
#include "../math/step.h"
|
||||||
|
|
||||||
|
namespace dsp::clock_recovery {
|
||||||
|
class FD : public Processor<float, float> {
|
||||||
|
using base_type = Processor<float, float> ;
|
||||||
|
public:
|
||||||
|
FD() {}
|
||||||
|
|
||||||
|
FD(stream<float>* in, double omega, double omegaGain, double muGain, double omegaRelLimit, int interpPhaseCount = 128, int interpTapCount = 8) { init(in, omega, omegaGain, muGain, omegaRelLimit, interpPhaseCount, interpTapCount); }
|
||||||
|
|
||||||
|
~FD() {
|
||||||
|
if (!base_type::_block_init) { return; }
|
||||||
|
base_type::stop();
|
||||||
|
dsp::multirate::freePolyphaseBank(interpBank);
|
||||||
|
buffer::free(buffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
void init(stream<float>* in, double omega, double omegaGain, double muGain, double omegaRelLimit, int interpPhaseCount = 128, int interpTapCount = 8) {
|
||||||
|
_omega = omega;
|
||||||
|
_omegaGain = omegaGain;
|
||||||
|
_muGain = muGain;
|
||||||
|
_omegaRelLimit = omegaRelLimit;
|
||||||
|
_interpPhaseCount = interpPhaseCount;
|
||||||
|
_interpTapCount = interpTapCount;
|
||||||
|
|
||||||
|
pcl.init(_muGain, _omegaGain, 0.0, 0.0, 1.0, _omega, _omega * (1.0 - omegaRelLimit), _omega * (1.0 + omegaRelLimit));
|
||||||
|
generateInterpTaps();
|
||||||
|
buffer = buffer::alloc<float>(STREAM_BUFFER_SIZE + _interpTapCount);
|
||||||
|
bufStart = &buffer[_interpTapCount - 1];
|
||||||
|
|
||||||
|
base_type::init(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
void setOmega(double omega) {
|
||||||
|
assert(base_type::_block_init);
|
||||||
|
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||||
|
base_type::tempStop();
|
||||||
|
_omega = omega;
|
||||||
|
offset = 0;
|
||||||
|
pcl.phase = 0.0f;
|
||||||
|
pcl.freq = _omega;
|
||||||
|
pcl.setFreqLimits(_omega * (1.0 - _omegaRelLimit), _omega * (1.0 + _omegaRelLimit));
|
||||||
|
base_type::tempStart();
|
||||||
|
}
|
||||||
|
|
||||||
|
void setOmegaGain(double omegaGain) {
|
||||||
|
assert(base_type::_block_init);
|
||||||
|
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||||
|
_omegaGain = omegaGain;
|
||||||
|
pcl.setCoefficients(_muGain, _omegaGain);
|
||||||
|
}
|
||||||
|
|
||||||
|
void setMuGain(double muGain) {
|
||||||
|
assert(base_type::_block_init);
|
||||||
|
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||||
|
_muGain = muGain;
|
||||||
|
pcl.setCoefficients(_muGain, _omegaGain);
|
||||||
|
}
|
||||||
|
|
||||||
|
void setOmegaRelLimit(double omegaRelLimit) {
|
||||||
|
assert(base_type::_block_init);
|
||||||
|
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||||
|
_omegaRelLimit = omegaRelLimit;
|
||||||
|
pcl.setFreqLimits(_omega * (1.0 - _omegaRelLimit), _omega * (1.0 + _omegaRelLimit));
|
||||||
|
}
|
||||||
|
|
||||||
|
void setInterpParams(int interpPhaseCount, int interpTapCount) {
|
||||||
|
assert(base_type::_block_init);
|
||||||
|
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||||
|
base_type::tempStop();
|
||||||
|
_interpPhaseCount = interpPhaseCount;
|
||||||
|
_interpTapCount = interpTapCount;
|
||||||
|
dsp::multirate::freePolyphaseBank(interpBank);
|
||||||
|
buffer::free(buffer);
|
||||||
|
generateInterpTaps();
|
||||||
|
buffer = buffer::alloc<float>(STREAM_BUFFER_SIZE + _interpTapCount);
|
||||||
|
bufStart = &buffer[_interpTapCount - 1];
|
||||||
|
base_type::tempStart();
|
||||||
|
}
|
||||||
|
|
||||||
|
void reset() {
|
||||||
|
assert(base_type::_block_init);
|
||||||
|
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||||
|
base_type::tempStop();
|
||||||
|
offset = 0;
|
||||||
|
pcl.phase = 0.0f;
|
||||||
|
pcl.freq = _omega;
|
||||||
|
base_type::tempStart();
|
||||||
|
}
|
||||||
|
|
||||||
|
inline int process(int count, const float* in, float* out) {
|
||||||
|
// Copy data to work buffer
|
||||||
|
memcpy(bufStart, in, count * sizeof(float));
|
||||||
|
|
||||||
|
// Process all samples
|
||||||
|
int outCount = 0;
|
||||||
|
while (offset < count) {
|
||||||
|
float error;
|
||||||
|
float outVal;
|
||||||
|
float dfdt;
|
||||||
|
|
||||||
|
// Calculate new output value
|
||||||
|
int phase = std::clamp<int>(floorf(pcl.phase * (float)_interpPhaseCount), 0, _interpPhaseCount - 1);
|
||||||
|
volk_32f_x2_dot_prod_32f(&outVal, &buffer[offset], interpBank.phases[phase], _interpTapCount);
|
||||||
|
out[outCount++] = outVal;
|
||||||
|
|
||||||
|
// Calculate derivative of the signal
|
||||||
|
if (phase == 0) {
|
||||||
|
float fT1;
|
||||||
|
volk_32f_x2_dot_prod_32f(&fT1, &buffer[offset], interpBank.phases[phase+1], _interpTapCount);
|
||||||
|
dfdt = fT1 - outVal;
|
||||||
|
}
|
||||||
|
else if (phase == _interpPhaseCount - 1) {
|
||||||
|
float fT_1;
|
||||||
|
volk_32f_x2_dot_prod_32f(&fT_1, &buffer[offset], interpBank.phases[phase-1], _interpTapCount);
|
||||||
|
dfdt = outVal - fT_1;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
float fT_1;
|
||||||
|
float fT1;
|
||||||
|
volk_32f_x2_dot_prod_32f(&fT_1, &buffer[offset], interpBank.phases[phase-1], _interpTapCount);
|
||||||
|
volk_32f_x2_dot_prod_32f(&fT1, &buffer[offset], interpBank.phases[phase+1], _interpTapCount);
|
||||||
|
dfdt = (fT1 - fT_1) * 0.5f;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Calculate error
|
||||||
|
error = dfdt * math::step(outVal);
|
||||||
|
|
||||||
|
// Clamp symbol phase error
|
||||||
|
if (error > 1.0f) { error = 1.0f; }
|
||||||
|
if (error < -1.0f) { error = -1.0f; }
|
||||||
|
|
||||||
|
// Advance symbol offset and phase
|
||||||
|
pcl.advance(error);
|
||||||
|
float delta = floorf(pcl.phase);
|
||||||
|
offset += delta;
|
||||||
|
pcl.phase -= delta;
|
||||||
|
}
|
||||||
|
offset -= count;
|
||||||
|
|
||||||
|
// Update delay buffer
|
||||||
|
memmove(buffer, &buffer[count], (_interpTapCount - 1) * sizeof(float));
|
||||||
|
|
||||||
|
return outCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
int run() {
|
||||||
|
int count = base_type::_in->read();
|
||||||
|
if (count < 0) { return -1; }
|
||||||
|
|
||||||
|
int outCount = process(count, base_type::_in->readBuf, base_type::out.writeBuf);
|
||||||
|
|
||||||
|
// Swap if some data was generated
|
||||||
|
base_type::_in->flush();
|
||||||
|
if (outCount) {
|
||||||
|
if (!base_type::out.swap(outCount)) { return -1; }
|
||||||
|
}
|
||||||
|
return outCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
loop::PhaseControlLoop<float, false> pcl;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void generateInterpTaps() {
|
||||||
|
double bw = 0.5 / (double)_interpPhaseCount;
|
||||||
|
dsp::tap<float> lp = dsp::taps::windowedSinc<float>(_interpPhaseCount * _interpTapCount, dsp::math::freqToOmega(bw, 1.0), dsp::window::nuttall, _interpPhaseCount);
|
||||||
|
interpBank = dsp::multirate::buildPolyphaseBank<float>(_interpPhaseCount, lp);
|
||||||
|
taps::free(lp);
|
||||||
|
}
|
||||||
|
|
||||||
|
dsp::multirate::PolyphaseBank<float> interpBank;
|
||||||
|
|
||||||
|
double _omega;
|
||||||
|
double _omegaGain;
|
||||||
|
double _muGain;
|
||||||
|
double _omegaRelLimit;
|
||||||
|
int _interpPhaseCount;
|
||||||
|
int _interpTapCount;
|
||||||
|
|
||||||
|
int offset = 0;
|
||||||
|
float* buffer;
|
||||||
|
float* bufStart;
|
||||||
|
};
|
||||||
|
}
|
@ -178,7 +178,7 @@ namespace dsp::clock_recovery {
|
|||||||
}
|
}
|
||||||
|
|
||||||
dsp::multirate::PolyphaseBank<float> interpBank;
|
dsp::multirate::PolyphaseBank<float> interpBank;
|
||||||
loop::PhaseControlLoop<double, false> pcl;
|
loop::PhaseControlLoop<float, false> pcl;
|
||||||
|
|
||||||
double _omega;
|
double _omega;
|
||||||
double _omegaGain;
|
double _omegaGain;
|
||||||
|
@ -12,6 +12,7 @@
|
|||||||
#include "../math/multiply.h"
|
#include "../math/multiply.h"
|
||||||
#include "../math/add.h"
|
#include "../math/add.h"
|
||||||
#include "../math/subtract.h"
|
#include "../math/subtract.h"
|
||||||
|
#include "../multirate/rational_resampler.h"
|
||||||
|
|
||||||
namespace dsp::demod {
|
namespace dsp::demod {
|
||||||
class BroadcastFM : public Processor<complex_t, stereo_t> {
|
class BroadcastFM : public Processor<complex_t, stereo_t> {
|
||||||
@ -19,7 +20,7 @@ namespace dsp::demod {
|
|||||||
public:
|
public:
|
||||||
BroadcastFM() {}
|
BroadcastFM() {}
|
||||||
|
|
||||||
BroadcastFM(stream<complex_t>* in, double deviation, double samplerate, bool stereo = true, bool lowPass = true) { init(in, deviation, samplerate, stereo, lowPass); }
|
BroadcastFM(stream<complex_t>* in, double deviation, double samplerate, bool stereo = true, bool lowPass = true, bool rdsOut = false) { init(in, deviation, samplerate, stereo, lowPass); }
|
||||||
|
|
||||||
~BroadcastFM() {
|
~BroadcastFM() {
|
||||||
if (!base_type::_block_init) { return; }
|
if (!base_type::_block_init) { return; }
|
||||||
@ -31,11 +32,12 @@ namespace dsp::demod {
|
|||||||
taps::free(audioFirTaps);
|
taps::free(audioFirTaps);
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void init(stream<complex_t>* in, double deviation, double samplerate, bool stereo = true, bool lowPass = true) {
|
virtual void init(stream<complex_t>* in, double deviation, double samplerate, bool stereo = true, bool lowPass = true, bool rdsOut = false) {
|
||||||
_deviation = deviation;
|
_deviation = deviation;
|
||||||
_samplerate = samplerate;
|
_samplerate = samplerate;
|
||||||
_stereo = stereo;
|
_stereo = stereo;
|
||||||
_lowPass = lowPass;
|
_lowPass = lowPass;
|
||||||
|
_rdsOut = rdsOut;
|
||||||
|
|
||||||
demod.init(NULL, _deviation, _samplerate);
|
demod.init(NULL, _deviation, _samplerate);
|
||||||
pilotFirTaps = taps::bandPass<complex_t>(18750.0, 19250.0, 3000.0, _samplerate, true);
|
pilotFirTaps = taps::bandPass<complex_t>(18750.0, 19250.0, 3000.0, _samplerate, true);
|
||||||
@ -47,6 +49,7 @@ namespace dsp::demod {
|
|||||||
audioFirTaps = taps::lowPass(15000.0, 4000.0, _samplerate);
|
audioFirTaps = taps::lowPass(15000.0, 4000.0, _samplerate);
|
||||||
alFir.init(NULL, audioFirTaps);
|
alFir.init(NULL, audioFirTaps);
|
||||||
arFir.init(NULL, audioFirTaps);
|
arFir.init(NULL, audioFirTaps);
|
||||||
|
rdsResamp.init(NULL, samplerate, 5000.0);
|
||||||
|
|
||||||
lmr = buffer::alloc<float>(STREAM_BUFFER_SIZE);
|
lmr = buffer::alloc<float>(STREAM_BUFFER_SIZE);
|
||||||
l = buffer::alloc<float>(STREAM_BUFFER_SIZE);
|
l = buffer::alloc<float>(STREAM_BUFFER_SIZE);
|
||||||
@ -56,6 +59,7 @@ namespace dsp::demod {
|
|||||||
lmrDelay.out.free();
|
lmrDelay.out.free();
|
||||||
arFir.out.free();
|
arFir.out.free();
|
||||||
alFir.out.free();
|
alFir.out.free();
|
||||||
|
rdsResamp.out.free();
|
||||||
|
|
||||||
base_type::init(in);
|
base_type::init(in);
|
||||||
}
|
}
|
||||||
@ -88,6 +92,8 @@ namespace dsp::demod {
|
|||||||
alFir.setTaps(audioFirTaps);
|
alFir.setTaps(audioFirTaps);
|
||||||
arFir.setTaps(audioFirTaps);
|
arFir.setTaps(audioFirTaps);
|
||||||
|
|
||||||
|
rdsResamp.setInSamplerate(samplerate);
|
||||||
|
|
||||||
reset();
|
reset();
|
||||||
base_type::tempStart();
|
base_type::tempStart();
|
||||||
}
|
}
|
||||||
@ -110,6 +116,15 @@ namespace dsp::demod {
|
|||||||
base_type::tempStart();
|
base_type::tempStart();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void setRDSOut(bool rdsOut) {
|
||||||
|
assert(base_type::_block_init);
|
||||||
|
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||||
|
base_type::tempStop();
|
||||||
|
_rdsOut = rdsOut;
|
||||||
|
reset();
|
||||||
|
base_type::tempStart();
|
||||||
|
}
|
||||||
|
|
||||||
void reset() {
|
void reset() {
|
||||||
assert(base_type::_block_init);
|
assert(base_type::_block_init);
|
||||||
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||||
@ -124,7 +139,7 @@ namespace dsp::demod {
|
|||||||
base_type::tempStart();
|
base_type::tempStart();
|
||||||
}
|
}
|
||||||
|
|
||||||
inline int process(int count, complex_t* in, stereo_t* out) {
|
inline int process(int count, complex_t* in, stereo_t* out, int& rdsOutCount, float* rdsout = NULL) {
|
||||||
// Demodulate
|
// Demodulate
|
||||||
demod.process(count, in, demod.out.writeBuf);
|
demod.process(count, in, demod.out.writeBuf);
|
||||||
if (_stereo) {
|
if (_stereo) {
|
||||||
@ -139,10 +154,19 @@ namespace dsp::demod {
|
|||||||
lprDelay.process(count, demod.out.writeBuf, demod.out.writeBuf);
|
lprDelay.process(count, demod.out.writeBuf, demod.out.writeBuf);
|
||||||
lmrDelay.process(count, rtoc.out.writeBuf, rtoc.out.writeBuf);
|
lmrDelay.process(count, rtoc.out.writeBuf, rtoc.out.writeBuf);
|
||||||
|
|
||||||
// Double and conjugate PLL output to down convert the L-R signal
|
// conjugate PLL output to down convert twice the L-R signal
|
||||||
math::Multiply<dsp::complex_t>::process(count, pilotPLL.out.writeBuf, pilotPLL.out.writeBuf, pilotPLL.out.writeBuf);
|
|
||||||
math::Conjugate::process(count, pilotPLL.out.writeBuf, pilotPLL.out.writeBuf);
|
math::Conjugate::process(count, pilotPLL.out.writeBuf, pilotPLL.out.writeBuf);
|
||||||
math::Multiply<dsp::complex_t>::process(count, rtoc.out.writeBuf, pilotPLL.out.writeBuf, rtoc.out.writeBuf);
|
math::Multiply<dsp::complex_t>::process(count, rtoc.out.writeBuf, pilotPLL.out.writeBuf, rtoc.out.writeBuf);
|
||||||
|
math::Multiply<dsp::complex_t>::process(count, rtoc.out.writeBuf, pilotPLL.out.writeBuf, rtoc.out.writeBuf);
|
||||||
|
|
||||||
|
// Do RDS demod
|
||||||
|
if (_rdsOut) {
|
||||||
|
// Since the PLL output is no longer needed after this, use it as the output
|
||||||
|
math::Multiply<dsp::complex_t>::process(count, rtoc.out.writeBuf, pilotPLL.out.writeBuf, pilotPLL.out.writeBuf);
|
||||||
|
convert::ComplexToReal::process(count, pilotPLL.out.writeBuf, rdsout);
|
||||||
|
volk_32f_s32f_multiply_32f(rdsout, rdsout, 100.0, count);
|
||||||
|
rdsOutCount = rdsResamp.process(count, rdsout, rdsout);
|
||||||
|
}
|
||||||
|
|
||||||
// Convert output back to real for further processing
|
// Convert output back to real for further processing
|
||||||
convert::ComplexToReal::process(count, rtoc.out.writeBuf, lmr);
|
convert::ComplexToReal::process(count, rtoc.out.writeBuf, lmr);
|
||||||
@ -180,18 +204,25 @@ namespace dsp::demod {
|
|||||||
int count = base_type::_in->read();
|
int count = base_type::_in->read();
|
||||||
if (count < 0) { return -1; }
|
if (count < 0) { return -1; }
|
||||||
|
|
||||||
process(count, base_type::_in->readBuf, base_type::out.writeBuf);
|
int rdsOutCount = 0;
|
||||||
|
process(count, base_type::_in->readBuf, base_type::out.writeBuf, rdsOutCount, rdsOut.writeBuf);
|
||||||
|
|
||||||
base_type::_in->flush();
|
base_type::_in->flush();
|
||||||
if (!base_type::out.swap(count)) { return -1; }
|
if (!base_type::out.swap(count)) { return -1; }
|
||||||
|
if (rdsOutCount && _rdsOut) {
|
||||||
|
if (!rdsOut.swap(rdsOutCount)) { return -1; }
|
||||||
|
}
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
stream<float> rdsOut;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
double _deviation;
|
double _deviation;
|
||||||
double _samplerate;
|
double _samplerate;
|
||||||
bool _stereo;
|
bool _stereo;
|
||||||
bool _lowPass = true;
|
bool _lowPass;
|
||||||
|
bool _rdsOut;
|
||||||
|
|
||||||
Quadrature demod;
|
Quadrature demod;
|
||||||
tap<complex_t> pilotFirTaps;
|
tap<complex_t> pilotFirTaps;
|
||||||
@ -203,6 +234,7 @@ namespace dsp::demod {
|
|||||||
tap<float> audioFirTaps;
|
tap<float> audioFirTaps;
|
||||||
filter::FIR<float, float> arFir;
|
filter::FIR<float, float> arFir;
|
||||||
filter::FIR<float, float> alFir;
|
filter::FIR<float, float> alFir;
|
||||||
|
multirate::RationalResampler<float> rdsResamp;
|
||||||
|
|
||||||
float* lmr;
|
float* lmr;
|
||||||
float* l;
|
float* l;
|
||||||
|
31
core/src/dsp/digital/binary_slicer.h
Normal file
31
core/src/dsp/digital/binary_slicer.h
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
#pragma once
|
||||||
|
#include "../processor.h"
|
||||||
|
|
||||||
|
namespace dsp::digital {
|
||||||
|
class BinarySlicer : public Processor<float, uint8_t> {
|
||||||
|
using base_type = Processor<float, uint8_t>;
|
||||||
|
public:
|
||||||
|
BinarySlicer() {}
|
||||||
|
|
||||||
|
BinarySlicer(stream<float> *in) { base_type::init(in); }
|
||||||
|
|
||||||
|
static inline int process(int count, const float* in, uint8_t* out) {
|
||||||
|
// TODO: Switch to volk
|
||||||
|
for (int i = 0; i < count; i++) {
|
||||||
|
out[i] = in[i] > 0.0f;
|
||||||
|
}
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
int run() {
|
||||||
|
int count = base_type::_in->read();
|
||||||
|
if (count < 0) { return -1; }
|
||||||
|
|
||||||
|
process(count, base_type::_in->readBuf, base_type::out.writeBuf);
|
||||||
|
|
||||||
|
base_type::_in->flush();
|
||||||
|
if (!base_type::out.swap(count)) { return -1; }
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
64
core/src/dsp/digital/differentia_decoder.h
Normal file
64
core/src/dsp/digital/differentia_decoder.h
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
#pragma once
|
||||||
|
#include "../processor.h"
|
||||||
|
|
||||||
|
namespace dsp::digital {
|
||||||
|
class DifferentialDecoder : public Processor<uint8_t, uint8_t> {
|
||||||
|
using base_type = Processor<uint8_t, uint8_t>;
|
||||||
|
public:
|
||||||
|
DifferentialDecoder() {}
|
||||||
|
|
||||||
|
DifferentialDecoder(stream<uint8_t> *in) { base_type::init(in); }
|
||||||
|
|
||||||
|
void init(stream<uint8_t> *in, uint8_t modulus, uint8_t initSym = 0) {
|
||||||
|
_modulus = modulus;
|
||||||
|
_initSym = initSym;
|
||||||
|
|
||||||
|
last = _initSym;
|
||||||
|
|
||||||
|
base_type::init(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
void setModulus(uint8_t modulus) {
|
||||||
|
assert(base_type::_block_init);
|
||||||
|
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||||
|
_modulus = modulus;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setInitSym(uint8_t initSym) {
|
||||||
|
assert(base_type::_block_init);
|
||||||
|
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||||
|
_initSym = initSym;
|
||||||
|
}
|
||||||
|
|
||||||
|
void reset() {
|
||||||
|
assert(base_type::_block_init);
|
||||||
|
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||||
|
base_type::tempStop();
|
||||||
|
last = _initSym;
|
||||||
|
base_type::tempStart();
|
||||||
|
}
|
||||||
|
|
||||||
|
inline int process(int count, const uint8_t* in, uint8_t* out) {
|
||||||
|
for (int i = 0; i < count; i++) {
|
||||||
|
out[i] = (in[i] - last) % _modulus;
|
||||||
|
}
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
int run() {
|
||||||
|
int count = base_type::_in->read();
|
||||||
|
if (count < 0) { return -1; }
|
||||||
|
|
||||||
|
process(count, base_type::_in->readBuf, base_type::out.writeBuf);
|
||||||
|
|
||||||
|
base_type::_in->flush();
|
||||||
|
if (!base_type::out.swap(count)) { return -1; }
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
uint8_t last;
|
||||||
|
uint8_t _initSym;
|
||||||
|
uint8_t _modulus;
|
||||||
|
};
|
||||||
|
}
|
65
core/src/dsp/digital/differential_decoder.h
Normal file
65
core/src/dsp/digital/differential_decoder.h
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
#pragma once
|
||||||
|
#include "../processor.h"
|
||||||
|
|
||||||
|
namespace dsp::digital {
|
||||||
|
class DifferentialDecoder : public Processor<uint8_t, uint8_t> {
|
||||||
|
using base_type = Processor<uint8_t, uint8_t>;
|
||||||
|
public:
|
||||||
|
DifferentialDecoder() {}
|
||||||
|
|
||||||
|
DifferentialDecoder(stream<uint8_t> *in) { base_type::init(in); }
|
||||||
|
|
||||||
|
void init(stream<uint8_t> *in, uint8_t modulus, uint8_t initSym = 0) {
|
||||||
|
_modulus = modulus;
|
||||||
|
_initSym = initSym;
|
||||||
|
|
||||||
|
last = _initSym;
|
||||||
|
|
||||||
|
base_type::init(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
void setModulus(uint8_t modulus) {
|
||||||
|
assert(base_type::_block_init);
|
||||||
|
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||||
|
_modulus = modulus;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setInitSym(uint8_t initSym) {
|
||||||
|
assert(base_type::_block_init);
|
||||||
|
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||||
|
_initSym = initSym;
|
||||||
|
}
|
||||||
|
|
||||||
|
void reset() {
|
||||||
|
assert(base_type::_block_init);
|
||||||
|
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||||
|
base_type::tempStop();
|
||||||
|
last = _initSym;
|
||||||
|
base_type::tempStart();
|
||||||
|
}
|
||||||
|
|
||||||
|
inline int process(int count, const uint8_t* in, uint8_t* out) {
|
||||||
|
for (int i = 0; i < count; i++) {
|
||||||
|
out[i] = (in[i] - last + _modulus) % _modulus;
|
||||||
|
last = in[i];
|
||||||
|
}
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
int run() {
|
||||||
|
int count = base_type::_in->read();
|
||||||
|
if (count < 0) { return -1; }
|
||||||
|
|
||||||
|
process(count, base_type::_in->readBuf, base_type::out.writeBuf);
|
||||||
|
|
||||||
|
base_type::_in->flush();
|
||||||
|
if (!base_type::out.swap(count)) { return -1; }
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
uint8_t last;
|
||||||
|
uint8_t _initSym;
|
||||||
|
uint8_t _modulus;
|
||||||
|
};
|
||||||
|
}
|
47
core/src/dsp/digital/manchester_decoder.h
Normal file
47
core/src/dsp/digital/manchester_decoder.h
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
#pragma once
|
||||||
|
#include "../processor.h"
|
||||||
|
|
||||||
|
namespace dsp::digital {
|
||||||
|
class ManchesterDecoder : public Processor<uint8_t, uint8_t> {
|
||||||
|
using base_type = Processor<uint8_t, uint8_t>;
|
||||||
|
public:
|
||||||
|
ManchesterDecoder() {}
|
||||||
|
|
||||||
|
ManchesterDecoder(stream<uint8_t> *in) { base_type::init(in); }
|
||||||
|
|
||||||
|
void reset() {
|
||||||
|
assert(base_type::_block_init);
|
||||||
|
std::lock_guard<std::recursive_mutex> lck(base_type::ctrlMtx);
|
||||||
|
base_type::tempStop();
|
||||||
|
offset = 0;
|
||||||
|
base_type::tempStart();
|
||||||
|
}
|
||||||
|
|
||||||
|
inline int process(int count, const uint8_t* in, uint8_t* out) {
|
||||||
|
// TODO: NOT THIS BULLSHIT
|
||||||
|
int outCount = 0;
|
||||||
|
for (; offset < count; offset += 2) {
|
||||||
|
out[outCount++] = in[offset];
|
||||||
|
}
|
||||||
|
offset -= count;
|
||||||
|
return outCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
int run() {
|
||||||
|
int count = base_type::_in->read();
|
||||||
|
if (count < 0) { return -1; }
|
||||||
|
|
||||||
|
int outCount = process(count, base_type::_in->readBuf, base_type::out.writeBuf);
|
||||||
|
|
||||||
|
// Swap if some data was generated
|
||||||
|
base_type::_in->flush();
|
||||||
|
if (outCount) {
|
||||||
|
if (!base_type::out.swap(outCount)) { return -1; }
|
||||||
|
}
|
||||||
|
return outCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
int offset = 0;
|
||||||
|
};
|
||||||
|
}
|
@ -1,13 +1,13 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include <imgui.h>
|
#include <imgui.h>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <module.h>
|
||||||
|
|
||||||
namespace style {
|
namespace style {
|
||||||
extern ImFont* baseFont;
|
SDRPP_EXPORT ImFont* baseFont;
|
||||||
extern ImFont* bigFont;
|
SDRPP_EXPORT ImFont* bigFont;
|
||||||
extern ImFont* hugeFont;
|
SDRPP_EXPORT ImFont* hugeFont;
|
||||||
|
SDRPP_EXPORT float uiScale;
|
||||||
extern float uiScale;
|
|
||||||
|
|
||||||
bool setDefaultStyle(std::string resDir);
|
bool setDefaultStyle(std::string resDir);
|
||||||
bool loadFonts(std::string resDir);
|
bool loadFonts(std::string resDir);
|
||||||
|
@ -56,7 +56,7 @@ public:
|
|||||||
config.release(created);
|
config.release(created);
|
||||||
|
|
||||||
vfo = sigpath::vfoManager.createVFO(name, ImGui::WaterfallVFO::REF_CENTER, 0, 150000, INPUT_SAMPLE_RATE, 150000, 150000, true);
|
vfo = sigpath::vfoManager.createVFO(name, ImGui::WaterfallVFO::REF_CENTER, 0, 150000, INPUT_SAMPLE_RATE, 150000, 150000, true);
|
||||||
demod.init(vfo->output, 72000.0f, INPUT_SAMPLE_RATE, 33, 0.6f, 0.1f, 0.005f, (0.01 * 0.01) / 4.0, 0.01);
|
demod.init(vfo->output, 72000.0f, INPUT_SAMPLE_RATE, 33, 0.6f, 0.1f, 0.005f, 1e-6, 0.01);
|
||||||
split.init(&demod.out);
|
split.init(&demod.out);
|
||||||
split.bindStream(&symSinkStream);
|
split.bindStream(&symSinkStream);
|
||||||
split.bindStream(&sinkStream);
|
split.bindStream(&sinkStream);
|
||||||
|
@ -1,6 +1,15 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include "../demod.h"
|
#include "../demod.h"
|
||||||
#include <dsp/demod/broadcast_fm.h>
|
#include <dsp/demod/broadcast_fm.h>
|
||||||
|
#include <dsp/clock_recovery/mm.h>
|
||||||
|
#include <dsp/clock_recovery/fd.h>
|
||||||
|
#include <dsp/taps/root_raised_cosine.h>
|
||||||
|
#include <dsp/digital/binary_slicer.h>
|
||||||
|
#include <dsp/digital/manchester_decoder.h>
|
||||||
|
#include <dsp/digital/differential_decoder.h>
|
||||||
|
#include <gui/widgets/symbol_diagram.h>
|
||||||
|
#include <fstream>
|
||||||
|
#include <rds.h>
|
||||||
|
|
||||||
namespace demod {
|
namespace demod {
|
||||||
class WFM : public Demodulator {
|
class WFM : public Demodulator {
|
||||||
@ -13,12 +22,17 @@ namespace demod {
|
|||||||
|
|
||||||
~WFM() {
|
~WFM() {
|
||||||
stop();
|
stop();
|
||||||
|
gui::waterfall.onFFTRedraw.unbindHandler(&fftRedrawHandler);
|
||||||
}
|
}
|
||||||
|
|
||||||
void init(std::string name, ConfigManager* config, dsp::stream<dsp::complex_t>* input, double bandwidth, double audioSR) {
|
void init(std::string name, ConfigManager* config, dsp::stream<dsp::complex_t>* input, double bandwidth, double audioSR) {
|
||||||
this->name = name;
|
this->name = name;
|
||||||
_config = config;
|
_config = config;
|
||||||
|
|
||||||
|
fftRedrawHandler.handler = fftRedraw;
|
||||||
|
fftRedrawHandler.ctx = this;
|
||||||
|
gui::waterfall.onFFTRedraw.bindHandler(&fftRedrawHandler);
|
||||||
|
|
||||||
// Load config
|
// Load config
|
||||||
_config->acquire();
|
_config->acquire();
|
||||||
bool modified = false;
|
bool modified = false;
|
||||||
@ -28,18 +42,36 @@ namespace demod {
|
|||||||
if (config->conf[name][getName()].contains("lowPass")) {
|
if (config->conf[name][getName()].contains("lowPass")) {
|
||||||
_lowPass = config->conf[name][getName()]["lowPass"];
|
_lowPass = config->conf[name][getName()]["lowPass"];
|
||||||
}
|
}
|
||||||
|
if (config->conf[name][getName()].contains("rds")) {
|
||||||
|
_rds = config->conf[name][getName()]["rds"];
|
||||||
|
}
|
||||||
_config->release(modified);
|
_config->release(modified);
|
||||||
|
|
||||||
// Define structure
|
// Define structure
|
||||||
demod.init(input, bandwidth / 2.0f, getIFSampleRate(), _stereo, _lowPass);
|
demod.init(input, bandwidth / 2.0f, getIFSampleRate(), _stereo, _lowPass, _rds);
|
||||||
|
recov.init(&demod.rdsOut, 5000.0 / 2375, omegaGain, muGain, 0.01);
|
||||||
|
slice.init(&recov.out);
|
||||||
|
manch.init(&slice.out);
|
||||||
|
diff.init(&manch.out, 2);
|
||||||
|
hs.init(&diff.out, rdsHandler, this);
|
||||||
}
|
}
|
||||||
|
|
||||||
void start() {
|
void start() {
|
||||||
demod.start();
|
demod.start();
|
||||||
|
recov.start();
|
||||||
|
slice.start();
|
||||||
|
manch.start();
|
||||||
|
diff.start();
|
||||||
|
hs.start();
|
||||||
}
|
}
|
||||||
|
|
||||||
void stop() {
|
void stop() {
|
||||||
demod.stop();
|
demod.stop();
|
||||||
|
recov.stop();
|
||||||
|
slice.stop();
|
||||||
|
manch.stop();
|
||||||
|
diff.stop();
|
||||||
|
hs.stop();
|
||||||
}
|
}
|
||||||
|
|
||||||
void showMenu() {
|
void showMenu() {
|
||||||
@ -55,6 +87,21 @@ namespace demod {
|
|||||||
_config->conf[name][getName()]["lowPass"] = _lowPass;
|
_config->conf[name][getName()]["lowPass"] = _lowPass;
|
||||||
_config->release(true);
|
_config->release(true);
|
||||||
}
|
}
|
||||||
|
if (ImGui::Checkbox(("Decode RDS##_radio_wfm_rds_" + name).c_str(), &_rds)) {
|
||||||
|
demod.setRDSOut(_rds);
|
||||||
|
_config->acquire();
|
||||||
|
_config->conf[name][getName()]["rds"] = _rds;
|
||||||
|
_config->release(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
// if (_rds) {
|
||||||
|
// if (rdsDecode.countryCodeValid()) { ImGui::Text("Country code: %d", rdsDecode.getCountryCode()); }
|
||||||
|
// if (rdsDecode.programCoverageValid()) { ImGui::Text("Program coverage: %d", rdsDecode.getProgramCoverage()); }
|
||||||
|
// if (rdsDecode.programRefNumberValid()) { ImGui::Text("Reference number: %d", rdsDecode.getProgramRefNumber()); }
|
||||||
|
// if (rdsDecode.programTypeValid()) { ImGui::Text("Program type: %d", rdsDecode.getProgramType()); }
|
||||||
|
// if (rdsDecode.PSNameValid()) { ImGui::Text("Program name: [%s]", rdsDecode.getPSName().c_str()); }
|
||||||
|
// if (rdsDecode.radioTextValid()) { ImGui::Text("Radiotext: [%s]", rdsDecode.getRadioText().c_str()); }
|
||||||
|
// }
|
||||||
}
|
}
|
||||||
|
|
||||||
void setBandwidth(double bandwidth) {
|
void setBandwidth(double bandwidth) {
|
||||||
@ -93,12 +140,68 @@ namespace demod {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
static void rdsHandler(uint8_t* data, int count, void* ctx) {
|
||||||
|
WFM* _this = (WFM*)ctx;
|
||||||
|
_this->rdsDecode.process(data, count);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void fftRedraw(ImGui::WaterFall::FFTRedrawArgs args, void* ctx) {
|
||||||
|
WFM* _this = (WFM*)ctx;
|
||||||
|
if (!_this->_rds) { return; }
|
||||||
|
|
||||||
|
// Generate string depending on RDS mode
|
||||||
|
char buf[256];
|
||||||
|
if (_this->rdsDecode.PSNameValid() && _this->rdsDecode.radioTextValid()) {
|
||||||
|
sprintf(buf, "RDS: %s - %s", _this->rdsDecode.getPSName().c_str(), _this->rdsDecode.getRadioText().c_str());
|
||||||
|
}
|
||||||
|
else if (_this->rdsDecode.PSNameValid()) {
|
||||||
|
sprintf(buf, "RDS: %s", _this->rdsDecode.getPSName().c_str());
|
||||||
|
}
|
||||||
|
else if (_this->rdsDecode.radioTextValid()) {
|
||||||
|
sprintf(buf, "RDS: %s", _this->rdsDecode.getRadioText().c_str());
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Calculate paddings
|
||||||
|
ImVec2 min = args.min;
|
||||||
|
min.x += 5.0f * style::uiScale;
|
||||||
|
min.y += 5.0f * style::uiScale;
|
||||||
|
ImVec2 tmin = min;
|
||||||
|
tmin.x += 5.0f * style::uiScale;
|
||||||
|
tmin.y += 5.0f * style::uiScale;
|
||||||
|
ImVec2 tmax = ImGui::CalcTextSize(buf);
|
||||||
|
tmax.x += tmin.x;
|
||||||
|
tmax.y += tmin.y;
|
||||||
|
ImVec2 max = tmax;
|
||||||
|
max.x += 5.0f * style::uiScale;
|
||||||
|
max.y += 5.0f * style::uiScale;
|
||||||
|
|
||||||
|
// Draw back drop
|
||||||
|
args.window->DrawList->AddRectFilled(min, max, IM_COL32(0, 0, 0, 128));
|
||||||
|
|
||||||
|
// Draw text
|
||||||
|
args.window->DrawList->AddText(tmin, IM_COL32(255, 255, 0, 255), buf);
|
||||||
|
}
|
||||||
|
|
||||||
dsp::demod::BroadcastFM demod;
|
dsp::demod::BroadcastFM demod;
|
||||||
|
dsp::clock_recovery::FD recov;
|
||||||
|
dsp::digital::BinarySlicer slice;
|
||||||
|
dsp::digital::ManchesterDecoder manch;
|
||||||
|
dsp::digital::DifferentialDecoder diff;
|
||||||
|
dsp::sink::Handler<uint8_t> hs;
|
||||||
|
EventHandler<ImGui::WaterFall::FFTRedrawArgs> fftRedrawHandler;
|
||||||
|
|
||||||
|
rds::RDSDecoder rdsDecode;
|
||||||
|
|
||||||
ConfigManager* _config = NULL;
|
ConfigManager* _config = NULL;
|
||||||
|
|
||||||
bool _stereo = false;
|
bool _stereo = false;
|
||||||
bool _lowPass = true;
|
bool _lowPass = true;
|
||||||
|
bool _rds = false;
|
||||||
|
float muGain = 0.01;
|
||||||
|
float omegaGain = (0.01*0.01)/4.0;
|
||||||
|
|
||||||
std::string name;
|
std::string name;
|
||||||
};
|
};
|
||||||
|
@ -232,16 +232,19 @@ private:
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Noise blanker
|
// Noise blanker
|
||||||
if (ImGui::Checkbox(("Noise blanker (W.I.P.)##_radio_nb_ena_" + _this->name).c_str(), &_this->nbEnabled)) {
|
if (_this->nbAllowed) {
|
||||||
_this->setNBEnabled(_this->nbEnabled);
|
if (ImGui::Checkbox(("Noise blanker (W.I.P.)##_radio_nb_ena_" + _this->name).c_str(), &_this->nbEnabled)) {
|
||||||
|
_this->setNBEnabled(_this->nbEnabled);
|
||||||
|
}
|
||||||
|
if (!_this->nbEnabled && _this->enabled) { style::beginDisabled(); }
|
||||||
|
ImGui::SameLine();
|
||||||
|
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
||||||
|
if (ImGui::SliderFloat(("##_radio_nb_lvl_" + _this->name).c_str(), &_this->nbLevel, _this->MIN_NB, _this->MAX_NB, "%.3fdB")) {
|
||||||
|
_this->setNBLevel(_this->nbLevel);
|
||||||
|
}
|
||||||
|
if (!_this->nbEnabled && _this->enabled) { style::endDisabled(); }
|
||||||
}
|
}
|
||||||
if (!_this->nbEnabled && _this->enabled) { style::beginDisabled(); }
|
|
||||||
ImGui::SameLine();
|
|
||||||
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
|
|
||||||
if (ImGui::SliderFloat(("##_radio_nb_lvl_" + _this->name).c_str(), &_this->nbLevel, _this->MIN_NB, _this->MAX_NB, "%.3fdB")) {
|
|
||||||
_this->setNBLevel(_this->nbLevel);
|
|
||||||
}
|
|
||||||
if (!_this->nbEnabled && _this->enabled) { style::endDisabled(); }
|
|
||||||
|
|
||||||
// Squelch
|
// Squelch
|
||||||
if (ImGui::Checkbox(("Squelch##_radio_sqelch_ena_" + _this->name).c_str(), &_this->squelchEnabled)) {
|
if (ImGui::Checkbox(("Squelch##_radio_sqelch_ena_" + _this->name).c_str(), &_this->squelchEnabled)) {
|
||||||
@ -422,7 +425,7 @@ private:
|
|||||||
// Configure noise blanker
|
// Configure noise blanker
|
||||||
nb.setRate(500.0 / ifSamplerate);
|
nb.setRate(500.0 / ifSamplerate);
|
||||||
setNBLevel(nbLevel);
|
setNBLevel(nbLevel);
|
||||||
setNBEnabled(nbEnabled);
|
setNBEnabled(nbAllowed&& nbEnabled);
|
||||||
|
|
||||||
// Configure FM IF Noise Reduction
|
// Configure FM IF Noise Reduction
|
||||||
setIFNRPreset((selectedDemodID == RADIO_DEMOD_NFM) ? ifnrPresets[fmIFPresetId] : IFNR_PRESET_BROADCAST);
|
setIFNRPreset((selectedDemodID == RADIO_DEMOD_NFM) ? ifnrPresets[fmIFPresetId] : IFNR_PRESET_BROADCAST);
|
||||||
|
251
decoder_modules/radio/src/rds.cpp
Normal file
251
decoder_modules/radio/src/rds.cpp
Normal file
@ -0,0 +1,251 @@
|
|||||||
|
#include "rds.h"
|
||||||
|
#include <string.h>
|
||||||
|
#include <map>
|
||||||
|
#include <algorithm>
|
||||||
|
|
||||||
|
namespace rds {
|
||||||
|
std::map<uint16_t, BlockType> SYNDROMES = {
|
||||||
|
{ 0b1111011000, BLOCK_TYPE_A },
|
||||||
|
{ 0b1111010100, BLOCK_TYPE_B },
|
||||||
|
{ 0b1001011100, BLOCK_TYPE_C },
|
||||||
|
{ 0b1111001100, BLOCK_TYPE_CP },
|
||||||
|
{ 0b1001011000, BLOCK_TYPE_D }
|
||||||
|
};
|
||||||
|
|
||||||
|
std::map<BlockType, uint16_t> OFFSETS = {
|
||||||
|
{ BLOCK_TYPE_A, 0b0011111100 },
|
||||||
|
{ BLOCK_TYPE_B, 0b0110011000 },
|
||||||
|
{ BLOCK_TYPE_C, 0b0101101000 },
|
||||||
|
{ BLOCK_TYPE_CP, 0b1101010000 },
|
||||||
|
{ BLOCK_TYPE_D, 0b0110110100 }
|
||||||
|
};
|
||||||
|
|
||||||
|
// This parity check matrix is given in annex B2.1 of the specificiation
|
||||||
|
const uint16_t PARITY_CHECK_MAT[] = {
|
||||||
|
0b1000000000,
|
||||||
|
0b0100000000,
|
||||||
|
0b0010000000,
|
||||||
|
0b0001000000,
|
||||||
|
0b0000100000,
|
||||||
|
0b0000010000,
|
||||||
|
0b0000001000,
|
||||||
|
0b0000000100,
|
||||||
|
0b0000000010,
|
||||||
|
0b0000000001,
|
||||||
|
0b1011011100,
|
||||||
|
0b0101101110,
|
||||||
|
0b0010110111,
|
||||||
|
0b1010000111,
|
||||||
|
0b1110011111,
|
||||||
|
0b1100010011,
|
||||||
|
0b1101010101,
|
||||||
|
0b1101110110,
|
||||||
|
0b0110111011,
|
||||||
|
0b1000000001,
|
||||||
|
0b1111011100,
|
||||||
|
0b0111101110,
|
||||||
|
0b0011110111,
|
||||||
|
0b1010100111,
|
||||||
|
0b1110001111,
|
||||||
|
0b1100011011
|
||||||
|
};
|
||||||
|
|
||||||
|
const int BLOCK_LEN = 26;
|
||||||
|
const int DATA_LEN = 16;
|
||||||
|
const int POLY_LEN = 10;
|
||||||
|
const uint16_t LFSR_POLY = 0x5B9;
|
||||||
|
const uint16_t IN_POLY = 0x31B;
|
||||||
|
|
||||||
|
void RDSDecoder::process(uint8_t* symbols, int count) {
|
||||||
|
for (int i = 0; i < count; i++) {
|
||||||
|
// Shift in the bit
|
||||||
|
shiftReg = ((shiftReg << 1) & 0x3FFFFFF) | (symbols[i] & 1);
|
||||||
|
|
||||||
|
// Skip if we need to shift in new data
|
||||||
|
if (--skip > 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Calculate the syndrome and update sync status
|
||||||
|
uint16_t syn = calcSyndrome(shiftReg);
|
||||||
|
auto synIt = SYNDROMES.find(syn);
|
||||||
|
bool knownSyndrome = synIt != SYNDROMES.end();
|
||||||
|
sync = std::clamp<int>(knownSyndrome ? ++sync : --sync, 0, 4);
|
||||||
|
|
||||||
|
// if (knownSyndrome) {
|
||||||
|
// printf("Found known syn: %04X\n", syn);
|
||||||
|
// }
|
||||||
|
|
||||||
|
// If we're still no longer in sync, try to resync
|
||||||
|
if (!sync) { continue; }
|
||||||
|
|
||||||
|
// Figure out which block we've got
|
||||||
|
BlockType type;
|
||||||
|
if (knownSyndrome) {
|
||||||
|
type = SYNDROMES[syn];
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
type = (BlockType)((lastType + 1) % _BLOCK_TYPE_COUNT);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Save block while correcting errors (NOT YET)
|
||||||
|
blocks[type] = shiftReg;//correctErrors(shiftReg, type);
|
||||||
|
|
||||||
|
//printf("Block type: %d, Sync: %d, KnownSyndrome: %s, contGroup: %d, offset: %d\n", type, sync, knownSyndrome ? "yes" : "no", contGroup, i);
|
||||||
|
|
||||||
|
// Update continous group count
|
||||||
|
if (type == BLOCK_TYPE_A) { contGroup = 1; }
|
||||||
|
else if (type == BLOCK_TYPE_B && lastType == BLOCK_TYPE_A) { contGroup++; }
|
||||||
|
else if ((type == BLOCK_TYPE_C || type == BLOCK_TYPE_CP) && lastType == BLOCK_TYPE_B) { contGroup++; }
|
||||||
|
else if (type == BLOCK_TYPE_D && (lastType == BLOCK_TYPE_C || lastType == BLOCK_TYPE_CP)) { contGroup++; }
|
||||||
|
else { contGroup = 0; }
|
||||||
|
|
||||||
|
// If we've got an entire group, process it
|
||||||
|
if (contGroup >= 4) {
|
||||||
|
contGroup = 0;
|
||||||
|
decodeGroup();
|
||||||
|
}
|
||||||
|
|
||||||
|
// // Remember the last block type and skip to new block
|
||||||
|
lastType = type;
|
||||||
|
skip = BLOCK_LEN;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16_t RDSDecoder::calcSyndrome(uint32_t block) {
|
||||||
|
// Perform vector/matrix dot product between block and parity matrix
|
||||||
|
uint16_t syn = 0;
|
||||||
|
for(int i = 0; i < BLOCK_LEN; i++) {
|
||||||
|
syn ^= PARITY_CHECK_MAT[BLOCK_LEN - 1 - i] * ((block >> i) & 1);
|
||||||
|
}
|
||||||
|
return syn;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32_t RDSDecoder::correctErrors(uint32_t block, BlockType type) {
|
||||||
|
// Init the syndrome and output
|
||||||
|
uint16_t syn = 0;
|
||||||
|
uint32_t out = block;
|
||||||
|
|
||||||
|
// Subtract the offset from block
|
||||||
|
block ^= (uint32_t)OFFSETS[type];
|
||||||
|
|
||||||
|
// Feed in the data
|
||||||
|
for (int i = BLOCK_LEN - 1; i >= 0; i--) {
|
||||||
|
// Shift the syndrome and keep the output
|
||||||
|
uint8_t outBit = (syn >> (POLY_LEN - 1)) & 1;
|
||||||
|
syn = (syn << 1) & 0b1111111111;
|
||||||
|
|
||||||
|
// Apply LFSR polynomial
|
||||||
|
syn ^= LFSR_POLY * outBit;
|
||||||
|
|
||||||
|
// Apply input polynomial.
|
||||||
|
syn ^= IN_POLY * ((block >> i) & 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Use the syndrome register to do error correction
|
||||||
|
// TODO: For some reason there's always zeros in the syn when starting
|
||||||
|
uint8_t errorFound = 0;
|
||||||
|
for (int i = DATA_LEN - 1; i >= 0; i--) {
|
||||||
|
// Check if the 5 leftmost bits are all zero
|
||||||
|
errorFound |= !(syn & 0b11111);
|
||||||
|
|
||||||
|
// Write output
|
||||||
|
uint8_t outBit = (syn >> (POLY_LEN - 1)) & 1;
|
||||||
|
out ^= (errorFound & outBit) << (i + POLY_LEN);
|
||||||
|
|
||||||
|
// Shift syndrome
|
||||||
|
syn = (syn << 1) & 0b1111111111;
|
||||||
|
syn ^= LFSR_POLY * outBit * !errorFound;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: mark block as irrecoverable if too damaged
|
||||||
|
|
||||||
|
if (errorFound) {
|
||||||
|
printf("Error found\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
|
void RDSDecoder::decodeGroup() {
|
||||||
|
std::lock_guard<std::mutex> lck(groupMtx);
|
||||||
|
auto now = std::chrono::high_resolution_clock::now();
|
||||||
|
anyGroupLastUpdate = now;
|
||||||
|
|
||||||
|
// Decode PI code
|
||||||
|
countryCode = (blocks[BLOCK_TYPE_A] >> 22) & 0xF;
|
||||||
|
programCoverage = (AreaCoverage)((blocks[BLOCK_TYPE_A] >> 18) & 0xF);
|
||||||
|
programRefNumber = (blocks[BLOCK_TYPE_A] >> 10) & 0xFF;
|
||||||
|
|
||||||
|
// Decode group type and version
|
||||||
|
uint8_t groupType = (blocks[BLOCK_TYPE_B] >> 22) & 0xF;
|
||||||
|
GroupVersion groupVer = (GroupVersion)((blocks[BLOCK_TYPE_B] >> 21) & 1);
|
||||||
|
|
||||||
|
// Decode traffic program and program type
|
||||||
|
trafficProgram = (blocks[BLOCK_TYPE_B] >> 20) & 1;
|
||||||
|
programType = (ProgramType)((blocks[BLOCK_TYPE_B] >> 15) & 0x1F);
|
||||||
|
|
||||||
|
if (groupType == 0) {
|
||||||
|
group0LastUpdate = now;
|
||||||
|
trafficAnnouncement = (blocks[BLOCK_TYPE_B] >> 14) & 1;
|
||||||
|
music = (blocks[BLOCK_TYPE_B] >> 13) & 1;
|
||||||
|
uint8_t diBit = (blocks[BLOCK_TYPE_B] >> 12) & 1;
|
||||||
|
uint8_t offset = ((blocks[BLOCK_TYPE_B] >> 10) & 0b11);
|
||||||
|
uint8_t diOffset = 3 - offset;
|
||||||
|
uint8_t psOffset = offset * 2;
|
||||||
|
|
||||||
|
if (groupVer == GROUP_VER_A) {
|
||||||
|
alternateFrequency = (blocks[BLOCK_TYPE_C] >> 10) & 0xFFFF;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write DI bit to the decoder identification
|
||||||
|
decoderIdent &= ~(1 << diOffset);
|
||||||
|
decoderIdent |= (diBit << diOffset);
|
||||||
|
|
||||||
|
// Write chars at offset the PSName
|
||||||
|
programServiceName[psOffset] = (blocks[BLOCK_TYPE_D] >> 18) & 0xFF;
|
||||||
|
programServiceName[psOffset + 1] = (blocks[BLOCK_TYPE_D] >> 10) & 0xFF;
|
||||||
|
}
|
||||||
|
else if (groupType == 2) {
|
||||||
|
group2LastUpdate = now;
|
||||||
|
// Get char offset and write chars in the Radiotext
|
||||||
|
bool nAB = (blocks[BLOCK_TYPE_B] >> 14) & 1;
|
||||||
|
uint8_t offset = (blocks[BLOCK_TYPE_B] >> 10) & 0xF;
|
||||||
|
|
||||||
|
// Clear text field if the A/B flag changed
|
||||||
|
if (nAB != rtAB) {
|
||||||
|
radioText = " ";
|
||||||
|
}
|
||||||
|
rtAB = nAB;
|
||||||
|
|
||||||
|
// Write char at offset in Radiotext
|
||||||
|
if (groupVer == GROUP_VER_A) {
|
||||||
|
uint8_t rtOffset = offset * 4;
|
||||||
|
radioText[rtOffset] = (blocks[BLOCK_TYPE_C] >> 18) & 0xFF;
|
||||||
|
radioText[rtOffset + 1] = (blocks[BLOCK_TYPE_C] >> 10) & 0xFF;
|
||||||
|
radioText[rtOffset + 2] = (blocks[BLOCK_TYPE_D] >> 18) & 0xFF;
|
||||||
|
radioText[rtOffset + 3] = (blocks[BLOCK_TYPE_D] >> 10) & 0xFF;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
uint8_t rtOffset = offset * 2;
|
||||||
|
radioText[rtOffset] = (blocks[BLOCK_TYPE_D] >> 18) & 0xFF;
|
||||||
|
radioText[rtOffset + 1] = (blocks[BLOCK_TYPE_D] >> 10) & 0xFF;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool RDSDecoder::anyGroupValid() {
|
||||||
|
auto now = std::chrono::high_resolution_clock::now();
|
||||||
|
return (std::chrono::duration_cast<std::chrono::milliseconds>(now - anyGroupLastUpdate)).count() < 5000.0;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool RDSDecoder::group0Valid() {
|
||||||
|
auto now = std::chrono::high_resolution_clock::now();
|
||||||
|
return (std::chrono::duration_cast<std::chrono::milliseconds>(now - group0LastUpdate)).count() < 5000.0;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool RDSDecoder::group2Valid() {
|
||||||
|
auto now = std::chrono::high_resolution_clock::now();
|
||||||
|
return (std::chrono::duration_cast<std::chrono::milliseconds>(now - group2LastUpdate)).count() < 5000.0;
|
||||||
|
}
|
||||||
|
}
|
179
decoder_modules/radio/src/rds.h
Normal file
179
decoder_modules/radio/src/rds.h
Normal file
@ -0,0 +1,179 @@
|
|||||||
|
#pragma once
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <string>
|
||||||
|
#include <chrono>
|
||||||
|
#include <mutex>
|
||||||
|
|
||||||
|
namespace rds {
|
||||||
|
enum BlockType {
|
||||||
|
BLOCK_TYPE_A,
|
||||||
|
BLOCK_TYPE_B,
|
||||||
|
BLOCK_TYPE_C,
|
||||||
|
BLOCK_TYPE_CP,
|
||||||
|
BLOCK_TYPE_D,
|
||||||
|
_BLOCK_TYPE_COUNT
|
||||||
|
};
|
||||||
|
|
||||||
|
enum GroupVersion {
|
||||||
|
GROUP_VER_A,
|
||||||
|
GROUP_VER_B
|
||||||
|
};
|
||||||
|
|
||||||
|
enum AreaCoverage {
|
||||||
|
AREA_COVERAGE_LOCAL,
|
||||||
|
AREA_COVERAGE_INTERNATIONAL,
|
||||||
|
AREA_COVERAGE_NATIONAL,
|
||||||
|
AREA_COVERAGE_SUPRA_NATIONAL,
|
||||||
|
AREA_COVERAGE_REGIONAL1,
|
||||||
|
AREA_COVERAGE_REGIONAL2,
|
||||||
|
AREA_COVERAGE_REGIONAL3,
|
||||||
|
AREA_COVERAGE_REGIONAL4,
|
||||||
|
AREA_COVERAGE_REGIONAL5,
|
||||||
|
AREA_COVERAGE_REGIONAL6,
|
||||||
|
AREA_COVERAGE_REGIONAL7,
|
||||||
|
AREA_COVERAGE_REGIONAL8,
|
||||||
|
AREA_COVERAGE_REGIONAL9,
|
||||||
|
AREA_COVERAGE_REGIONAL10,
|
||||||
|
AREA_COVERAGE_REGIONAL11,
|
||||||
|
AREA_COVERAGE_REGIONAL12
|
||||||
|
};
|
||||||
|
|
||||||
|
enum ProgramType {
|
||||||
|
// US Types
|
||||||
|
PROGRAM_TYPE_US_NONE = 0,
|
||||||
|
PROGRAM_TYPE_US_NEWS = 1,
|
||||||
|
PROGRAM_TYPE_US_INFOMATION = 2,
|
||||||
|
PROGRAM_TYPE_US_SPORTS = 3,
|
||||||
|
PROGRAM_TYPE_US_TALK = 4,
|
||||||
|
PROGRAM_TYPE_US_ROCK = 5,
|
||||||
|
PROGRAM_TYPE_US_CLASSIC_ROCK = 6,
|
||||||
|
PROGRAM_TYPE_US_ADULT_HITS = 7,
|
||||||
|
PROGRAM_TYPE_US_SOFT_ROCK = 8,
|
||||||
|
PROGRAM_TYPE_US_TOP_40 = 9,
|
||||||
|
PROGRAM_TYPE_US_COUNTRY = 10,
|
||||||
|
PROGRAM_TYPE_US_OLDIES = 11,
|
||||||
|
PROGRAM_TYPE_US_SOFT = 12,
|
||||||
|
PROGRAM_TYPE_US_NOSTALGIA = 13,
|
||||||
|
PROGRAM_TYPE_US_JAZZ = 14,
|
||||||
|
PROGRAM_TYPE_US_CLASSICAL = 15,
|
||||||
|
PROGRAM_TYPE_US_RHYTHM_AND_BLUES = 16,
|
||||||
|
PROGRAM_TYPE_US_SOFT_RHYTHM_AND_BLUES = 17,
|
||||||
|
PROGRAM_TYPE_US_FOREIGN_LANGUAGE = 18,
|
||||||
|
PROGRAM_TYPE_US_RELIGIOUS_MUSIC = 19,
|
||||||
|
PROGRAM_TYPE_US_RELIGIOUS_TALK = 20,
|
||||||
|
PROGRAM_TYPE_US_PERSONALITY = 21,
|
||||||
|
PROGRAM_TYPE_US_PUBLIC = 22,
|
||||||
|
PROGRAM_TYPE_US_COLLEGE = 23,
|
||||||
|
PROGRAM_TYPE_US_UNASSIGNED0 = 24,
|
||||||
|
PROGRAM_TYPE_US_UNASSIGNED1 = 25,
|
||||||
|
PROGRAM_TYPE_US_UNASSIGNED2 = 26,
|
||||||
|
PROGRAM_TYPE_US_UNASSIGNED3 = 27,
|
||||||
|
PROGRAM_TYPE_US_UNASSIGNED4 = 28,
|
||||||
|
PROGRAM_TYPE_US_WEATHER = 29,
|
||||||
|
PROGRAM_TYPE_US_EMERGENCY_TEST = 30,
|
||||||
|
PROGRAM_TYPE_US_EMERGENCY = 31,
|
||||||
|
|
||||||
|
// EU Types
|
||||||
|
PROGRAM_TYPE_EU_NONE = 0,
|
||||||
|
PROGRAM_TYPE_EU_NEWS = 1,
|
||||||
|
PROGRAM_TYPE_EU_CURRENT_AFFAIRS = 2,
|
||||||
|
PROGRAM_TYPE_EU_INFORMATION = 3,
|
||||||
|
PROGRAM_TYPE_EU_SPORTS = 4,
|
||||||
|
PROGRAM_TYPE_EU_EDUCATION = 5,
|
||||||
|
PROGRAM_TYPE_EU_DRAMA = 6,
|
||||||
|
PROGRAM_TYPE_EU_CULTURE = 7,
|
||||||
|
PROGRAM_TYPE_EU_SCIENCE = 8,
|
||||||
|
PROGRAM_TYPE_EU_VARIED = 9,
|
||||||
|
PROGRAM_TYPE_EU_POP_MUSIC = 10,
|
||||||
|
PROGRAM_TYPE_EU_ROCK_MUSIC = 11,
|
||||||
|
PROGRAM_TYPE_EU_EASY_LISTENING_MUSIC = 12,
|
||||||
|
PROGRAM_TYPE_EU_LIGHT_CLASSICAL = 13,
|
||||||
|
PROGRAM_TYPE_EU_SERIOUS_CLASSICAL = 14,
|
||||||
|
PROGRAM_TYPE_EU_OTHER_MUSIC = 15,
|
||||||
|
PROGRAM_TYPE_EU_WEATHER = 16,
|
||||||
|
PROGRAM_TYPE_EU_FINANCE = 17,
|
||||||
|
PROGRAM_TYPE_EU_CHILDRENS_PROGRAM = 18,
|
||||||
|
PROGRAM_TYPE_EU_SOCIAL_AFFAIRS = 19,
|
||||||
|
PROGRAM_TYPE_EU_RELIGION = 20,
|
||||||
|
PROGRAM_TYPE_EU_PHONE_IN = 21,
|
||||||
|
PROGRAM_TYPE_EU_TRAVEL = 22,
|
||||||
|
PROGRAM_TYPE_EU_LEISURE = 23,
|
||||||
|
PROGRAM_TYPE_EU_JAZZ_MUSIC = 24,
|
||||||
|
PROGRAM_TYPE_EU_COUNTRY_MUSIC = 25,
|
||||||
|
PROGRAM_TYPE_EU_NATIONAL_MUSIC = 26,
|
||||||
|
PROGRAM_TYPE_EU_OLDIES_MUSIC = 27,
|
||||||
|
PROGRAM_TYPE_EU_FOLK_MUSIC = 28,
|
||||||
|
PROGRAM_TYPE_EU_DOCUMENTARY = 29,
|
||||||
|
PROGRAM_TYPE_EU_ALARM_TEST = 30,
|
||||||
|
PROGRAM_TYPE_EU_ALARM = 31
|
||||||
|
};
|
||||||
|
|
||||||
|
enum DecoderIdentification {
|
||||||
|
DECODER_IDENT_STEREO = (1 << 0),
|
||||||
|
DECODER_IDENT_ARTIFICIAL_HEAD = (1 << 1),
|
||||||
|
DECODER_IDENT_COMPRESSED = (1 << 2),
|
||||||
|
DECODER_IDENT_VARIABLE_PTY = (1 << 0)
|
||||||
|
};
|
||||||
|
|
||||||
|
class RDSDecoder {
|
||||||
|
public:
|
||||||
|
void process(uint8_t* symbols, int count);
|
||||||
|
|
||||||
|
bool countryCodeValid() { std::lock_guard<std::mutex> lck(groupMtx); return anyGroupValid(); }
|
||||||
|
uint8_t getCountryCode() { std::lock_guard<std::mutex> lck(groupMtx); return countryCode; }
|
||||||
|
bool programCoverageValid() { std::lock_guard<std::mutex> lck(groupMtx); return anyGroupValid(); }
|
||||||
|
uint8_t getProgramCoverage() { std::lock_guard<std::mutex> lck(groupMtx); return programCoverage; }
|
||||||
|
bool programRefNumberValid() { std::lock_guard<std::mutex> lck(groupMtx); return anyGroupValid(); }
|
||||||
|
uint8_t getProgramRefNumber() { std::lock_guard<std::mutex> lck(groupMtx); return programRefNumber; }
|
||||||
|
bool programTypeValid() { std::lock_guard<std::mutex> lck(groupMtx); return anyGroupValid(); }
|
||||||
|
ProgramType getProgramType() { std::lock_guard<std::mutex> lck(groupMtx); return programType; }
|
||||||
|
|
||||||
|
bool musicValid() { std::lock_guard<std::mutex> lck(groupMtx); return group0Valid(); }
|
||||||
|
bool getMusic() { std::lock_guard<std::mutex> lck(groupMtx); return music; }
|
||||||
|
bool PSNameValid() { std::lock_guard<std::mutex> lck(groupMtx); return group0Valid(); }
|
||||||
|
std::string getPSName() { std::lock_guard<std::mutex> lck(groupMtx); return programServiceName; }
|
||||||
|
|
||||||
|
bool radioTextValid() { std::lock_guard<std::mutex> lck(groupMtx); return group2Valid(); }
|
||||||
|
std::string getRadioText() { std::lock_guard<std::mutex> lck(groupMtx); return radioText; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
static uint16_t calcSyndrome(uint32_t block);
|
||||||
|
static uint32_t correctErrors(uint32_t block, BlockType type);
|
||||||
|
void decodeGroup();
|
||||||
|
|
||||||
|
bool anyGroupValid();
|
||||||
|
bool group0Valid();
|
||||||
|
bool group2Valid();
|
||||||
|
|
||||||
|
// State machine
|
||||||
|
uint32_t shiftReg = 0;
|
||||||
|
int sync = 0;
|
||||||
|
int skip = 0;
|
||||||
|
BlockType lastType = BLOCK_TYPE_A;
|
||||||
|
int contGroup = 0;
|
||||||
|
uint32_t blocks[_BLOCK_TYPE_COUNT];
|
||||||
|
|
||||||
|
// All groups
|
||||||
|
std::mutex groupMtx;
|
||||||
|
std::chrono::steady_clock::time_point anyGroupLastUpdate;
|
||||||
|
uint8_t countryCode;
|
||||||
|
AreaCoverage programCoverage;
|
||||||
|
uint8_t programRefNumber;
|
||||||
|
bool trafficProgram;
|
||||||
|
ProgramType programType;
|
||||||
|
|
||||||
|
// Group type 0
|
||||||
|
std::chrono::steady_clock::time_point group0LastUpdate;
|
||||||
|
bool trafficAnnouncement;
|
||||||
|
bool music;
|
||||||
|
uint8_t decoderIdent;
|
||||||
|
uint16_t alternateFrequency;
|
||||||
|
std::string programServiceName = " ";
|
||||||
|
|
||||||
|
// Group type 2
|
||||||
|
std::chrono::steady_clock::time_point group2LastUpdate;
|
||||||
|
bool rtAB = false;
|
||||||
|
std::string radioText = " ";
|
||||||
|
|
||||||
|
};
|
||||||
|
}
|
@ -56,10 +56,10 @@ cp $build_dir/sink_modules/network_sink/Release/network_sink.dll sdrpp_windows_x
|
|||||||
|
|
||||||
|
|
||||||
# Copy decoder modules
|
# Copy decoder modules
|
||||||
# cp $build_dir/decoder_modules/m17_decoder/Release/m17_decoder.dll sdrpp_windows_x64/modules/
|
cp $build_dir/decoder_modules/m17_decoder/Release/m17_decoder.dll sdrpp_windows_x64/modules/
|
||||||
# cp "C:/Program Files/codec2/lib/libcodec2.dll" sdrpp_windows_x64/
|
cp "C:/Program Files/codec2/lib/libcodec2.dll" sdrpp_windows_x64/
|
||||||
|
|
||||||
# cp $build_dir/decoder_modules/meteor_demodulator/Release/meteor_demodulator.dll sdrpp_windows_x64/modules/
|
cp $build_dir/decoder_modules/meteor_demodulator/Release/meteor_demodulator.dll sdrpp_windows_x64/modules/
|
||||||
cp $build_dir/decoder_modules/radio/Release/radio.dll sdrpp_windows_x64/modules/
|
cp $build_dir/decoder_modules/radio/Release/radio.dll sdrpp_windows_x64/modules/
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user