2017-01-02 21:07:43 -05:00
|
|
|
// Copyright (c) Charles J. Cliffe
|
|
|
|
// SPDX-License-Identifier: GPL-2.0+
|
|
|
|
|
2014-12-16 18:27:32 -05:00
|
|
|
#include "CubicSDRDefs.h"
|
2015-02-22 01:01:28 -05:00
|
|
|
#include "DemodulatorThread.h"
|
2015-11-17 23:23:23 -05:00
|
|
|
#include "DemodulatorInstance.h"
|
2016-02-11 01:09:15 -05:00
|
|
|
#include "CubicSDR.h"
|
2014-12-16 18:27:32 -05:00
|
|
|
#include <vector>
|
|
|
|
|
2015-02-22 01:01:28 -05:00
|
|
|
#include <cmath>
|
|
|
|
#ifndef M_PI
|
|
|
|
#define M_PI 3.14159265358979323846
|
|
|
|
#endif
|
|
|
|
|
2014-12-16 18:27:32 -05:00
|
|
|
#ifdef __APPLE__
|
|
|
|
#include <pthread.h>
|
|
|
|
#endif
|
|
|
|
|
2016-08-12 22:58:33 -04:00
|
|
|
std::atomic<DemodulatorInstance *> DemodulatorThread::squelchLock(nullptr);
|
|
|
|
std::mutex DemodulatorThread::squelchLockMutex;
|
|
|
|
|
|
|
|
DemodulatorThread::DemodulatorThread(DemodulatorInstance *parent)
|
2016-06-01 13:32:22 -04:00
|
|
|
: IOThread(), outputBuffers("DemodulatorThreadBuffers"), squelchLevel(-100),
|
2016-08-10 14:02:25 -04:00
|
|
|
signalLevel(-100), signalFloor(-30), signalCeil(30), squelchEnabled(false) {
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2015-11-17 23:23:23 -05:00
|
|
|
demodInstance = parent;
|
2015-08-17 00:59:38 -04:00
|
|
|
muted.store(false);
|
2016-02-11 01:32:39 -05:00
|
|
|
squelchBreak = false;
|
2014-12-16 18:27:32 -05:00
|
|
|
}
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2014-12-16 18:27:32 -05:00
|
|
|
DemodulatorThread::~DemodulatorThread() {
|
2016-08-13 14:18:41 -04:00
|
|
|
releaseSquelchLock(demodInstance);
|
2014-12-16 18:27:32 -05:00
|
|
|
}
|
|
|
|
|
2015-07-30 00:52:00 -04:00
|
|
|
void DemodulatorThread::onBindOutput(std::string name, ThreadQueueBase *threadQueue) {
|
|
|
|
if (name == "AudioVisualOutput") {
|
2016-06-01 13:32:22 -04:00
|
|
|
|
|
|
|
//protects because it may be changed at runtime
|
|
|
|
std::lock_guard < std::mutex > lock(m_mutexAudioVisOutputQueue);
|
|
|
|
|
2016-06-01 13:42:11 -04:00
|
|
|
audioVisOutputQueue = static_cast<DemodulatorThreadOutputQueue*>(threadQueue);
|
2015-07-30 00:52:00 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-13 15:12:20 -04:00
|
|
|
double DemodulatorThread::abMagnitude(float inphase, float quadrature) {
|
|
|
|
|
|
|
|
// cast to double, so we keep precision despite the **2 op later.
|
|
|
|
double dinphase = (double)inphase;
|
|
|
|
double dquadrature = (double)quadrature;
|
|
|
|
|
|
|
|
//sqrt() has been an insanely fast intrinsic for years, use it !
|
|
|
|
return sqrt(dinphase * dinphase + dquadrature * dquadrature);
|
|
|
|
|
2015-11-21 15:12:20 -05:00
|
|
|
}
|
|
|
|
|
2016-08-13 15:12:20 -04:00
|
|
|
double DemodulatorThread::linearToDb(double linear) {
|
|
|
|
|
2015-11-21 15:12:20 -05:00
|
|
|
#define SMALL 1e-20
|
|
|
|
if (linear <= SMALL) {
|
2016-08-13 15:12:20 -04:00
|
|
|
linear = double(SMALL);
|
2015-11-21 15:12:20 -05:00
|
|
|
}
|
|
|
|
return 20.0 * log10(linear);
|
|
|
|
}
|
|
|
|
|
2015-07-29 20:57:02 -04:00
|
|
|
void DemodulatorThread::run() {
|
2014-12-16 20:33:44 -05:00
|
|
|
#ifdef __APPLE__
|
2014-12-18 20:11:25 -05:00
|
|
|
pthread_t tID = pthread_self(); // ID of this thread
|
|
|
|
int priority = sched_get_priority_max( SCHED_FIFO )-1;
|
2014-12-18 21:39:32 -05:00
|
|
|
sched_param prio = {priority}; // scheduling priority of thread
|
2014-12-18 20:11:25 -05:00
|
|
|
pthread_setschedparam(tID, SCHED_FIFO, &prio);
|
2014-12-16 20:33:44 -05:00
|
|
|
#endif
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2016-06-02 19:45:34 -04:00
|
|
|
// std::cout << "Demodulator thread started.." << std::endl;
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2016-06-01 13:42:11 -04:00
|
|
|
iqInputQueue = static_cast<DemodulatorThreadPostInputQueue*>(getInputQueue("IQDataInput"));
|
|
|
|
audioOutputQueue = static_cast<AudioThreadInputQueue*>(getOutputQueue("AudioDataOutput"));
|
|
|
|
threadQueueControl = static_cast<DemodulatorThreadControlCommandQueue *>(getInputQueue("ControlQueue"));
|
2016-07-03 03:47:28 -04:00
|
|
|
|
2015-11-17 21:22:51 -05:00
|
|
|
ModemIQData modemData;
|
2015-07-30 00:28:53 -04:00
|
|
|
|
2016-06-28 15:04:52 -04:00
|
|
|
while (!stopping) {
|
2014-12-23 01:12:14 -05:00
|
|
|
DemodulatorThreadPostIQData *inp;
|
2016-07-06 15:23:59 -04:00
|
|
|
|
2015-01-03 17:07:39 -05:00
|
|
|
iqInputQueue->pop(inp);
|
2015-11-18 21:13:04 -05:00
|
|
|
// std::lock_guard < std::mutex > lock(inp->m_mutex);
|
|
|
|
|
2016-02-04 18:05:33 -05:00
|
|
|
size_t bufSize = inp->data.size();
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2014-12-16 21:30:03 -05:00
|
|
|
if (!bufSize) {
|
2014-12-24 01:28:33 -05:00
|
|
|
inp->decRefCount();
|
2014-12-16 21:30:03 -05:00
|
|
|
continue;
|
|
|
|
}
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2015-11-17 21:22:51 -05:00
|
|
|
if (inp->modemKit && inp->modemKit != cModemKit) {
|
2015-11-17 23:23:23 -05:00
|
|
|
if (cModemKit != nullptr) {
|
|
|
|
cModem->disposeKit(cModemKit);
|
|
|
|
}
|
2015-11-17 21:22:51 -05:00
|
|
|
cModemKit = inp->modemKit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (inp->modem && inp->modem != cModem) {
|
|
|
|
delete cModem;
|
|
|
|
cModem = inp->modem;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!cModem || !cModemKit) {
|
|
|
|
inp->decRefCount();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-07-19 15:34:06 -04:00
|
|
|
std::vector<liquid_float_complex> *inputData;
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2015-11-21 15:12:20 -05:00
|
|
|
inputData = &inp->data;
|
2015-11-17 21:22:51 -05:00
|
|
|
|
|
|
|
modemData.sampleRate = inp->sampleRate;
|
|
|
|
modemData.data.assign(inputData->begin(), inputData->end());
|
2015-11-17 23:23:23 -05:00
|
|
|
|
2016-06-01 13:32:22 -04:00
|
|
|
AudioThreadInput *ati = nullptr;
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2015-11-20 20:46:25 -05:00
|
|
|
ModemAnalog *modemAnalog = (cModem->getType() == "analog")?((ModemAnalog *)cModem):nullptr;
|
2015-11-29 13:35:12 -05:00
|
|
|
ModemDigital *modemDigital = (cModem->getType() == "digital")?((ModemDigital *)cModem):nullptr;
|
2015-11-20 20:46:25 -05:00
|
|
|
|
|
|
|
if (modemAnalog != nullptr) {
|
2015-11-19 00:11:34 -05:00
|
|
|
ati = outputBuffers.getBuffer();
|
|
|
|
|
2015-11-22 23:25:45 -05:00
|
|
|
ati->sampleRate = cModemKit->audioSampleRate;
|
2015-11-19 00:11:34 -05:00
|
|
|
ati->inputRate = inp->sampleRate;
|
2015-11-29 13:35:12 -05:00
|
|
|
} else if (modemDigital != nullptr) {
|
|
|
|
ati = outputBuffers.getBuffer();
|
|
|
|
|
|
|
|
ati->sampleRate = cModemKit->sampleRate;
|
|
|
|
ati->inputRate = inp->sampleRate;
|
2016-07-24 15:25:17 -04:00
|
|
|
ati->data.resize(0);
|
2015-11-19 00:11:34 -05:00
|
|
|
}
|
2015-11-29 13:35:12 -05:00
|
|
|
|
2015-11-17 21:22:51 -05:00
|
|
|
cModem->demodulate(cModemKit, &modemData, ati);
|
2016-08-12 20:31:20 -04:00
|
|
|
|
2016-08-13 15:12:20 -04:00
|
|
|
double currentSignalLevel = 0;
|
|
|
|
double sampleTime = double(inp->data.size()) / double(inp->sampleRate);
|
2016-08-12 20:31:20 -04:00
|
|
|
|
|
|
|
if (audioOutputQueue != nullptr && ati && ati->data.size()) {
|
2016-08-13 15:12:20 -04:00
|
|
|
double accum = 0;
|
|
|
|
|
|
|
|
if (cModem->useSignalOutput()) {
|
2016-08-12 21:34:34 -04:00
|
|
|
|
2016-08-13 14:50:20 -04:00
|
|
|
for (auto i : ati->data) {
|
2016-08-13 15:12:20 -04:00
|
|
|
accum += abMagnitude(i, 0.0);
|
2016-08-13 14:50:20 -04:00
|
|
|
}
|
2016-08-12 21:34:34 -04:00
|
|
|
|
2016-08-13 15:12:20 -04:00
|
|
|
currentSignalLevel = linearToDb(accum / double(ati->data.size()));
|
2016-08-12 21:34:34 -04:00
|
|
|
|
2016-08-13 14:50:20 -04:00
|
|
|
} else {
|
2016-08-13 15:12:20 -04:00
|
|
|
|
2016-08-13 14:50:20 -04:00
|
|
|
for (auto i : inp->data) {
|
2016-08-13 15:12:20 -04:00
|
|
|
accum += abMagnitude(i.real, i.imag);
|
2016-08-13 14:50:20 -04:00
|
|
|
}
|
2016-08-12 21:34:34 -04:00
|
|
|
|
2016-08-13 15:12:20 -04:00
|
|
|
currentSignalLevel = linearToDb(accum / double(inp->data.size()));
|
2016-08-13 14:50:20 -04:00
|
|
|
}
|
2016-08-12 20:31:20 -04:00
|
|
|
|
|
|
|
float sf = signalFloor.load(), sc = signalCeil.load(), sl = squelchLevel.load();
|
|
|
|
|
2016-08-13 15:12:20 -04:00
|
|
|
|
2016-08-12 20:31:20 -04:00
|
|
|
if (currentSignalLevel > sc) {
|
|
|
|
sc = currentSignalLevel;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (currentSignalLevel < sf) {
|
|
|
|
sf = currentSignalLevel;
|
|
|
|
}
|
|
|
|
|
2016-08-13 15:12:20 -04:00
|
|
|
|
2016-08-12 20:31:20 -04:00
|
|
|
if (sl+1.0f > sc) {
|
|
|
|
sc = sl+1.0f;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((sf+2.0f) > sc) {
|
|
|
|
sc = sf+2.0f;
|
|
|
|
}
|
|
|
|
|
2016-08-12 21:34:34 -04:00
|
|
|
sc -= (sc - (currentSignalLevel + 2.0f)) * sampleTime * 0.05f;
|
2016-08-12 20:31:20 -04:00
|
|
|
sf += ((currentSignalLevel - 5.0f) - sf) * sampleTime * 0.15f;
|
|
|
|
|
|
|
|
signalFloor.store(sf);
|
|
|
|
signalCeil.store(sc);
|
|
|
|
}
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2015-01-03 17:07:39 -05:00
|
|
|
if (currentSignalLevel > signalLevel) {
|
2016-08-12 21:34:34 -04:00
|
|
|
signalLevel = signalLevel + (currentSignalLevel - signalLevel) * 0.5;
|
2014-12-31 19:45:01 -05:00
|
|
|
} else {
|
2016-08-12 21:34:34 -04:00
|
|
|
signalLevel = signalLevel + (currentSignalLevel - signalLevel) * 0.05 * sampleTime * 30.0;
|
2014-12-31 19:45:01 -05:00
|
|
|
}
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2016-08-12 22:58:33 -04:00
|
|
|
bool squelched = (muted.load() || (squelchEnabled && (signalLevel < squelchLevel)));
|
2015-11-21 15:12:20 -05:00
|
|
|
|
2016-02-11 01:32:39 -05:00
|
|
|
if (squelchEnabled) {
|
|
|
|
if (!squelched && !squelchBreak) {
|
2016-08-16 17:55:40 -04:00
|
|
|
if (wxGetApp().getSoloMode() && !wxGetApp().getAppFrame()->isUserDemodBusy()) {
|
2016-08-12 22:58:33 -04:00
|
|
|
std::lock_guard < std::mutex > lock(squelchLockMutex);
|
|
|
|
if (squelchLock.load() == nullptr) {
|
|
|
|
squelchLock.store(demodInstance);
|
|
|
|
wxGetApp().getDemodMgr().setActiveDemodulator(nullptr);
|
|
|
|
wxGetApp().getDemodMgr().setActiveDemodulator(demodInstance, false);
|
|
|
|
squelchBreak = true;
|
|
|
|
demodInstance->getVisualCue()->triggerSquelchBreak(120);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
squelchBreak = true;
|
|
|
|
demodInstance->getVisualCue()->triggerSquelchBreak(120);
|
|
|
|
}
|
|
|
|
|
2016-02-11 01:32:39 -05:00
|
|
|
} else if (squelched && squelchBreak) {
|
2016-08-13 14:18:41 -04:00
|
|
|
releaseSquelchLock(demodInstance);
|
2016-02-11 01:32:39 -05:00
|
|
|
squelchBreak = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-24 15:25:17 -04:00
|
|
|
if (audioOutputQueue != nullptr && ati && ati->data.size() && !squelched) {
|
2015-11-21 15:12:20 -05:00
|
|
|
std::vector<float>::iterator data_i;
|
|
|
|
ati->peak = 0;
|
2016-08-12 21:34:34 -04:00
|
|
|
for (auto data_i : ati->data) {
|
|
|
|
float p = fabs(data_i);
|
2015-11-21 15:12:20 -05:00
|
|
|
if (p > ati->peak) {
|
|
|
|
ati->peak = p;
|
2015-02-08 18:59:08 -05:00
|
|
|
}
|
2014-12-21 16:08:32 -05:00
|
|
|
}
|
2015-11-21 15:12:20 -05:00
|
|
|
} else if (ati) {
|
2016-06-07 21:12:30 -04:00
|
|
|
ati->setRefCount(0);
|
2015-11-21 15:12:20 -05:00
|
|
|
ati = nullptr;
|
2014-12-16 21:30:03 -05:00
|
|
|
}
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2016-06-01 13:32:22 -04:00
|
|
|
//At that point, capture the current state of audioVisOutputQueue in a local
|
|
|
|
//variable, and works with it with now on until the next while-turn.
|
|
|
|
DemodulatorThreadOutputQueue* localAudioVisOutputQueue = nullptr;
|
|
|
|
{
|
|
|
|
std::lock_guard < std::mutex > lock(m_mutexAudioVisOutputQueue);
|
|
|
|
localAudioVisOutputQueue = audioVisOutputQueue;
|
|
|
|
}
|
|
|
|
|
2016-07-24 17:24:29 -04:00
|
|
|
if ((ati || modemDigital) && localAudioVisOutputQueue != nullptr && localAudioVisOutputQueue->empty()) {
|
2016-06-02 19:25:52 -04:00
|
|
|
AudioThreadInput *ati_vis = new AudioThreadInput;
|
2016-06-01 13:32:22 -04:00
|
|
|
|
2015-08-24 01:31:37 -04:00
|
|
|
ati_vis->sampleRate = inp->sampleRate;
|
|
|
|
ati_vis->inputRate = inp->sampleRate;
|
|
|
|
|
2016-01-28 15:49:31 -05:00
|
|
|
size_t num_vis = DEMOD_VIS_SIZE;
|
2015-11-29 13:35:12 -05:00
|
|
|
if (modemDigital) {
|
2016-07-24 17:24:29 -04:00
|
|
|
if (ati) { // TODO: handle digital modems with audio output
|
|
|
|
ati->setRefCount(0);
|
|
|
|
ati = nullptr;
|
|
|
|
}
|
2015-11-29 13:35:12 -05:00
|
|
|
ati_vis->data.resize(inputData->size());
|
|
|
|
ati_vis->channels = 2;
|
|
|
|
for (int i = 0, iMax = inputData->size() / 2; i < iMax; i++) {
|
|
|
|
ati_vis->data[i * 2] = (*inputData)[i].real;
|
|
|
|
ati_vis->data[i * 2 + 1] = (*inputData)[i].imag;
|
|
|
|
}
|
|
|
|
ati_vis->type = 2;
|
|
|
|
} else if (ati->channels==2) {
|
2014-12-27 15:04:43 -05:00
|
|
|
ati_vis->channels = 2;
|
|
|
|
int stereoSize = ati->data.size();
|
2015-06-27 23:23:43 -04:00
|
|
|
if (stereoSize > DEMOD_VIS_SIZE * 2) {
|
|
|
|
stereoSize = DEMOD_VIS_SIZE * 2;
|
2014-12-26 22:20:50 -05:00
|
|
|
}
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2014-12-26 22:20:50 -05:00
|
|
|
ati_vis->data.resize(stereoSize);
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2015-11-29 13:35:12 -05:00
|
|
|
if (inp->modemName == "I/Q") {
|
2015-06-27 23:23:43 -04:00
|
|
|
for (int i = 0; i < stereoSize / 2; i++) {
|
2015-11-21 15:12:20 -05:00
|
|
|
ati_vis->data[i] = (*inputData)[i].real * 0.75;
|
|
|
|
ati_vis->data[i + stereoSize / 2] = (*inputData)[i].imag * 0.75;
|
2015-06-27 23:23:43 -04:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (int i = 0; i < stereoSize / 2; i++) {
|
2015-11-22 23:25:45 -05:00
|
|
|
ati_vis->inputRate = cModemKit->audioSampleRate;
|
2015-08-24 01:31:37 -04:00
|
|
|
ati_vis->sampleRate = 36000;
|
2015-06-27 23:23:43 -04:00
|
|
|
ati_vis->data[i] = ati->data[i * 2];
|
|
|
|
ati_vis->data[i + stereoSize / 2] = ati->data[i * 2 + 1];
|
|
|
|
}
|
2014-12-21 16:08:32 -05:00
|
|
|
}
|
2015-11-29 13:35:12 -05:00
|
|
|
ati_vis->type = 1;
|
2014-12-18 21:39:32 -05:00
|
|
|
} else {
|
2016-01-28 15:49:31 -05:00
|
|
|
size_t numAudioWritten = ati->data.size();
|
2014-12-27 15:04:43 -05:00
|
|
|
ati_vis->channels = 1;
|
2015-11-20 20:46:25 -05:00
|
|
|
std::vector<float> *demodOutData = (modemAnalog != nullptr)?modemAnalog->getDemodOutputData():nullptr;
|
|
|
|
if ((numAudioWritten > bufSize) || (demodOutData == nullptr)) {
|
2015-11-22 23:25:45 -05:00
|
|
|
ati_vis->inputRate = cModemKit->audioSampleRate;
|
2015-11-20 20:46:25 -05:00
|
|
|
if (num_vis > numAudioWritten) {
|
|
|
|
num_vis = numAudioWritten;
|
|
|
|
}
|
|
|
|
ati_vis->data.assign(ati->data.begin(), ati->data.begin() + num_vis);
|
|
|
|
} else {
|
|
|
|
if (num_vis > demodOutData->size()) {
|
|
|
|
num_vis = demodOutData->size();
|
|
|
|
}
|
|
|
|
ati_vis->data.assign(demodOutData->begin(), demodOutData->begin() + num_vis);
|
2015-11-18 21:13:04 -05:00
|
|
|
}
|
2015-11-29 13:35:12 -05:00
|
|
|
ati_vis->type = 0;
|
2014-12-21 16:08:32 -05:00
|
|
|
}
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2017-02-15 14:27:57 -05:00
|
|
|
if (!localAudioVisOutputQueue->try_push(ati_vis)) {
|
|
|
|
//non-blocking push for audio-out
|
|
|
|
ati_vis->setRefCount(0);
|
|
|
|
std::cout << "DemodulatorThread::run() cannot push ati_vis into localAudioVisOutputQueue, is full !" << std::endl;
|
|
|
|
std::this_thread::yield();
|
|
|
|
}
|
2014-12-21 16:08:32 -05:00
|
|
|
}
|
2017-02-09 13:12:12 -05:00
|
|
|
|
2016-06-01 13:32:22 -04:00
|
|
|
if (ati != nullptr) {
|
2016-02-11 01:09:15 -05:00
|
|
|
if (!muted.load() && (!wxGetApp().getSoloMode() || (demodInstance == wxGetApp().getDemodMgr().getLastActiveDemodulator()))) {
|
2017-02-15 14:27:57 -05:00
|
|
|
|
2017-02-09 13:12:12 -05:00
|
|
|
audioOutputQueue->push(ati);
|
2015-08-17 00:59:38 -04:00
|
|
|
} else {
|
|
|
|
ati->setRefCount(0);
|
|
|
|
}
|
2015-07-19 15:34:06 -04:00
|
|
|
}
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2016-07-05 15:43:45 -04:00
|
|
|
DemodulatorThreadControlCommand command;
|
|
|
|
|
|
|
|
//empty command queue, execute commands
|
|
|
|
while (threadQueueControl->try_pop(command)) {
|
|
|
|
|
|
|
|
switch (command.cmd) {
|
|
|
|
case DemodulatorThreadControlCommand::DEMOD_THREAD_CMD_CTL_SQUELCH_ON:
|
|
|
|
squelchEnabled = true;
|
|
|
|
break;
|
|
|
|
case DemodulatorThreadControlCommand::DEMOD_THREAD_CMD_CTL_SQUELCH_OFF:
|
|
|
|
squelchEnabled = false;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2014-12-18 21:39:32 -05:00
|
|
|
}
|
2014-12-16 21:30:03 -05:00
|
|
|
}
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2016-07-05 15:43:45 -04:00
|
|
|
|
2014-12-24 01:28:33 -05:00
|
|
|
inp->decRefCount();
|
2014-12-16 21:30:03 -05:00
|
|
|
}
|
2016-06-28 15:04:52 -04:00
|
|
|
// end while !stopping
|
2015-11-17 21:22:51 -05:00
|
|
|
|
2016-07-05 15:43:45 -04:00
|
|
|
// Purge any unused inputs, with a non-blocking pop
|
|
|
|
DemodulatorThreadPostIQData *ref;
|
|
|
|
while (iqInputQueue->try_pop(ref)) {
|
|
|
|
|
2016-06-08 19:48:46 -04:00
|
|
|
if (ref) { // May have other consumers; just decrement
|
|
|
|
ref->decRefCount();
|
2016-06-07 21:12:30 -04:00
|
|
|
}
|
2016-06-07 20:36:59 -04:00
|
|
|
}
|
2016-07-05 15:43:45 -04:00
|
|
|
|
|
|
|
AudioThreadInput *ref_audio;
|
|
|
|
while (audioOutputQueue->try_pop(ref_audio)) {
|
|
|
|
|
|
|
|
if (ref_audio) { // Originated here; set RefCount to 0
|
|
|
|
ref_audio->setRefCount(0);
|
2016-06-07 21:12:30 -04:00
|
|
|
}
|
2016-06-07 20:36:59 -04:00
|
|
|
}
|
2016-07-05 15:43:45 -04:00
|
|
|
|
2015-07-28 21:56:39 -04:00
|
|
|
outputBuffers.purge();
|
2015-11-18 21:13:04 -05:00
|
|
|
|
2016-06-02 19:45:34 -04:00
|
|
|
// std::cout << "Demodulator thread done." << std::endl;
|
2015-01-03 17:07:39 -05:00
|
|
|
}
|
|
|
|
|
2014-12-16 18:27:32 -05:00
|
|
|
void DemodulatorThread::terminate() {
|
2016-06-28 15:04:52 -04:00
|
|
|
IOThread::terminate();
|
2014-12-23 01:12:14 -05:00
|
|
|
DemodulatorThreadPostIQData *inp = new DemodulatorThreadPostIQData; // push dummy to nudge queue
|
2017-02-09 13:12:12 -05:00
|
|
|
|
|
|
|
//VSO: blocking push
|
|
|
|
iqInputQueue->push(inp);
|
2014-12-16 18:27:32 -05:00
|
|
|
}
|
2014-12-26 20:58:42 -05:00
|
|
|
|
2015-08-17 00:59:38 -04:00
|
|
|
bool DemodulatorThread::isMuted() {
|
|
|
|
return muted.load();
|
|
|
|
}
|
|
|
|
|
|
|
|
void DemodulatorThread::setMuted(bool muted) {
|
|
|
|
this->muted.store(muted);
|
|
|
|
}
|
|
|
|
|
2014-12-31 19:45:01 -05:00
|
|
|
float DemodulatorThread::getSignalLevel() {
|
2015-07-19 15:34:06 -04:00
|
|
|
return signalLevel.load();
|
2014-12-31 19:45:01 -05:00
|
|
|
}
|
|
|
|
|
2016-08-10 14:02:25 -04:00
|
|
|
float DemodulatorThread::getSignalFloor() {
|
|
|
|
return signalFloor.load();
|
|
|
|
}
|
|
|
|
|
|
|
|
float DemodulatorThread::getSignalCeil() {
|
|
|
|
return signalCeil.load();
|
|
|
|
}
|
|
|
|
|
2014-12-31 19:45:01 -05:00
|
|
|
void DemodulatorThread::setSquelchLevel(float signal_level_in) {
|
2015-01-03 17:07:39 -05:00
|
|
|
if (!squelchEnabled) {
|
|
|
|
squelchEnabled = true;
|
2014-12-31 19:45:01 -05:00
|
|
|
}
|
2015-01-03 17:07:39 -05:00
|
|
|
squelchLevel = signal_level_in;
|
2014-12-31 19:45:01 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
float DemodulatorThread::getSquelchLevel() {
|
2015-01-03 17:07:39 -05:00
|
|
|
return squelchLevel;
|
2014-12-31 19:45:01 -05:00
|
|
|
}
|
2016-08-13 14:18:41 -04:00
|
|
|
|
|
|
|
bool DemodulatorThread::getSquelchBreak() {
|
|
|
|
return squelchBreak;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DemodulatorThread::releaseSquelchLock(DemodulatorInstance *inst) {
|
|
|
|
std::lock_guard < std::mutex > lock(squelchLockMutex);
|
2016-08-16 17:55:40 -04:00
|
|
|
if (inst == nullptr || squelchLock.load() == inst) {
|
2016-08-13 14:18:41 -04:00
|
|
|
squelchLock.store(nullptr);
|
|
|
|
}
|
|
|
|
}
|