mirror of
https://github.com/f4exb/sdrangel.git
synced 2024-11-16 05:11:49 -05:00
480 lines
12 KiB
C++
480 lines
12 KiB
C++
|
///////////////////////////////////////////////////////////////////////////////////
|
||
|
// Copyright (C) 2012 maintech GmbH, Otto-Hahn-Str. 15, 97204 Hoechberg, Germany //
|
||
|
// written by Christian Daniel //
|
||
|
// //
|
||
|
// 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 as version 3 of the License, or //
|
||
|
// //
|
||
|
// 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 V3 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/>. //
|
||
|
///////////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
#include <string.h>
|
||
|
#include <errno.h>
|
||
|
#include "osmosdrinput.h"
|
||
|
#include "osmosdrthread.h"
|
||
|
#include "osmosdrgui.h"
|
||
|
#include "util/simpleserializer.h"
|
||
|
|
||
|
MESSAGE_CLASS_DEFINITION(OsmoSDRInput::MsgConfigureOsmoSDR, Message)
|
||
|
|
||
|
OsmoSDRInput::Settings::Settings() :
|
||
|
m_swapIQ(false),
|
||
|
m_decimation(3),
|
||
|
m_lnaGain(-50),
|
||
|
m_mixerGain(40),
|
||
|
m_mixerEnhancement(0),
|
||
|
m_if1gain(-30),
|
||
|
m_if2gain(0),
|
||
|
m_if3gain(0),
|
||
|
m_if4gain(0),
|
||
|
m_if5gain(30),
|
||
|
m_if6gain(30),
|
||
|
m_opAmpI1(0),
|
||
|
m_opAmpI2(0),
|
||
|
m_opAmpQ1(0),
|
||
|
m_opAmpQ2(0),
|
||
|
m_dcOfsI(0),
|
||
|
m_dcOfsQ(0)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
void OsmoSDRInput::Settings::resetToDefaults()
|
||
|
{
|
||
|
m_swapIQ = false;
|
||
|
m_decimation = 3;
|
||
|
m_lnaGain = -50;
|
||
|
m_mixerGain = 40;
|
||
|
m_mixerEnhancement = 0;
|
||
|
m_if1gain = -30;
|
||
|
m_if2gain = 0;
|
||
|
m_if3gain = 0;
|
||
|
m_if4gain = 0;
|
||
|
m_if5gain = 30;
|
||
|
m_if6gain = 30;
|
||
|
m_opAmpI1 = 0;
|
||
|
m_opAmpI2 = 0;
|
||
|
m_opAmpQ1 = 0;
|
||
|
m_opAmpQ2 = 0;
|
||
|
m_dcOfsI = 0;
|
||
|
m_dcOfsQ = 0;
|
||
|
}
|
||
|
|
||
|
QByteArray OsmoSDRInput::Settings::serialize() const
|
||
|
{
|
||
|
SimpleSerializer s(1);
|
||
|
s.writeBool(1, m_swapIQ);
|
||
|
s.writeS32(2, m_decimation);
|
||
|
s.writeS32(3, m_lnaGain);
|
||
|
s.writeS32(4, m_mixerGain);
|
||
|
s.writeS32(5, m_mixerEnhancement);
|
||
|
s.writeS32(6, m_if1gain);
|
||
|
s.writeS32(7, m_if2gain);
|
||
|
s.writeS32(8, m_if3gain);
|
||
|
s.writeS32(9, m_if4gain);
|
||
|
s.writeS32(10, m_if5gain);
|
||
|
s.writeS32(11, m_if6gain);
|
||
|
s.writeS32(12, m_opAmpI1);
|
||
|
s.writeS32(13, m_opAmpI2);
|
||
|
s.writeS32(14, m_opAmpQ1);
|
||
|
s.writeS32(15, m_opAmpQ2);
|
||
|
s.writeS32(16, m_dcOfsI);
|
||
|
s.writeS32(17, m_dcOfsQ);
|
||
|
return s.final();
|
||
|
}
|
||
|
|
||
|
bool OsmoSDRInput::Settings::deserialize(const QByteArray& data)
|
||
|
{
|
||
|
SimpleDeserializer d(data);
|
||
|
|
||
|
if(!d.isValid()) {
|
||
|
resetToDefaults();
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if(d.getVersion() == 1) {
|
||
|
d.readBool(1, &m_swapIQ, false);
|
||
|
d.readS32(2, &m_decimation, 3);
|
||
|
d.readS32(3, &m_lnaGain, -50);
|
||
|
d.readS32(4, &m_mixerGain, 40);
|
||
|
d.readS32(5, &m_mixerEnhancement, 0);
|
||
|
d.readS32(6, &m_if1gain, -30);
|
||
|
d.readS32(7, &m_if2gain, 0);
|
||
|
d.readS32(8, &m_if3gain, 0);
|
||
|
d.readS32(9, &m_if4gain, 0);
|
||
|
d.readS32(10, &m_if5gain, 30);
|
||
|
d.readS32(11, &m_if6gain, 30);
|
||
|
d.readS32(12, &m_opAmpI1, 0);
|
||
|
d.readS32(13, &m_opAmpI2, 0);
|
||
|
d.readS32(14, &m_opAmpQ1, 0);
|
||
|
d.readS32(15, &m_opAmpQ2, 0);
|
||
|
d.readS32(16, &m_dcOfsI, 0);
|
||
|
d.readS32(17, &m_dcOfsQ, 0);
|
||
|
return true;
|
||
|
} else {
|
||
|
resetToDefaults();
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
#if 0
|
||
|
OsmoSDRInput::Settings::Settings() :
|
||
|
{
|
||
|
}
|
||
|
|
||
|
QString OsmoSDRInput::Settings::serialize() const
|
||
|
{
|
||
|
return QString("osmosdr:a:%1:%2:%3:%4:%5:%6:%7:%8:%9:%10:%11:%12:%13:%14:%15:%16:%17:%18")
|
||
|
.arg(centerFrequency)
|
||
|
.arg(swapIQ ? 1 : 0)
|
||
|
.arg(decimation)
|
||
|
.arg(lnaGain)
|
||
|
.arg(mixerGain)
|
||
|
.arg(mixerEnhancement)
|
||
|
.arg(if1gain)
|
||
|
.arg(if2gain)
|
||
|
.arg(if3gain)
|
||
|
.arg(if4gain)
|
||
|
.arg(if5gain)
|
||
|
.arg(if6gain)
|
||
|
.arg(opAmpI1)
|
||
|
.arg(opAmpI2)
|
||
|
.arg(opAmpQ1)
|
||
|
.arg(opAmpQ2)
|
||
|
.arg(dcOfsI)
|
||
|
.arg(dcOfsQ);
|
||
|
}
|
||
|
|
||
|
bool OsmoSDRInput::Settings::deserialize(const QString& settings)
|
||
|
{
|
||
|
QStringList list = settings.split(":");
|
||
|
if(list.size() < 2)
|
||
|
return false;
|
||
|
if(list[0] != "osmosdr")
|
||
|
return false;
|
||
|
|
||
|
if(list[1] == "a") {
|
||
|
bool ok;
|
||
|
if(list.size() != 20)
|
||
|
return false;
|
||
|
centerFrequency = list[2].toLongLong(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
swapIQ = (list[3].toInt(&ok) != 0) ? true : false;
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
decimation = list[4].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
lnaGain = list[5].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
mixerGain = list[6].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
mixerEnhancement = list[7].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
if1gain = list[8].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
if2gain = list[9].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
if3gain = list[10].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
if4gain = list[11].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
if5gain = list[12].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
if6gain = list[13].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
opAmpI1 = list[14].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
opAmpI2 = list[15].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
opAmpQ1 = list[16].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
opAmpQ2 = list[17].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
dcOfsI = list[18].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
dcOfsQ = list[19].toInt(&ok);
|
||
|
if(!ok)
|
||
|
return false;
|
||
|
return true;
|
||
|
} else {
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
MessageRegistrator OsmoSDRInput::MsgConfigureSourceOsmoSDR::ID("MsgConfigureSourceOsmoSDR");
|
||
|
#endif
|
||
|
OsmoSDRInput::OsmoSDRInput(MessageQueue* msgQueueToGUI) :
|
||
|
SampleSource(msgQueueToGUI),
|
||
|
m_settings(),
|
||
|
m_dev(NULL),
|
||
|
m_osmoSDRThread(NULL),
|
||
|
m_deviceDescription()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
OsmoSDRInput::~OsmoSDRInput()
|
||
|
{
|
||
|
stopInput();
|
||
|
}
|
||
|
|
||
|
bool OsmoSDRInput::startInput(int device)
|
||
|
{
|
||
|
QMutexLocker mutexLocker(&m_mutex);
|
||
|
|
||
|
if(m_dev != NULL)
|
||
|
stopInput();
|
||
|
|
||
|
char vendor[256];
|
||
|
char product[256];
|
||
|
char serial[256];
|
||
|
int res;
|
||
|
|
||
|
if(!m_sampleFifo.setSize(524288)) {
|
||
|
qCritical("Could not allocate SampleFifo");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if((res = osmosdr_open(&m_dev, device)) < 0) {
|
||
|
qCritical("could not open OsmoSDR #%d: %s", device, strerror(errno));
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
vendor[0] = '\0';
|
||
|
product[0] = '\0';
|
||
|
serial[0] = '\0';
|
||
|
if((res = osmosdr_get_usb_strings(m_dev, vendor, product, serial)) < 0) {
|
||
|
qCritical("error accessing USB device");
|
||
|
goto failed;
|
||
|
}
|
||
|
qDebug("OsmoSDRInput open: %s %s, SN: %s", vendor, product, serial);
|
||
|
m_deviceDescription = QString("%1 (SN %2)").arg(product).arg(serial);
|
||
|
|
||
|
if((res = osmosdr_set_tuner_gain_mode(m_dev, 1)) < 0) {
|
||
|
qCritical("error setting tuner gain mode");
|
||
|
goto failed;
|
||
|
}
|
||
|
|
||
|
if((res = osmosdr_reset_buffer(m_dev)) < 0) {
|
||
|
qCritical("could not reset USB EP buffers: %s", strerror(errno));
|
||
|
goto failed;
|
||
|
}
|
||
|
|
||
|
if((m_osmoSDRThread = new OsmoSDRThread(m_dev, &m_sampleFifo)) == NULL) {
|
||
|
qFatal("out of memory");
|
||
|
goto failed;
|
||
|
}
|
||
|
m_osmoSDRThread->startWork();
|
||
|
|
||
|
mutexLocker.unlock();
|
||
|
applySettings(m_generalSettings, m_settings, true);
|
||
|
|
||
|
qDebug("OsmoSDRInput: start");
|
||
|
|
||
|
return true;
|
||
|
|
||
|
failed:
|
||
|
stopInput();
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
void OsmoSDRInput::stopInput()
|
||
|
{
|
||
|
QMutexLocker mutexLocker(&m_mutex);
|
||
|
|
||
|
if(m_osmoSDRThread != NULL) {
|
||
|
m_osmoSDRThread->stopWork();
|
||
|
delete m_osmoSDRThread;
|
||
|
m_osmoSDRThread = NULL;
|
||
|
}
|
||
|
if(m_dev != NULL) {
|
||
|
osmosdr_close(m_dev);
|
||
|
m_dev = NULL;
|
||
|
}
|
||
|
m_deviceDescription.clear();
|
||
|
}
|
||
|
|
||
|
const QString& OsmoSDRInput::getDeviceDescription() const
|
||
|
{
|
||
|
return m_deviceDescription;
|
||
|
}
|
||
|
|
||
|
int OsmoSDRInput::getSampleRate() const
|
||
|
{
|
||
|
return 4000000 / (1 << m_settings.m_decimation);
|
||
|
}
|
||
|
|
||
|
quint64 OsmoSDRInput::getCenterFrequency() const
|
||
|
{
|
||
|
return m_generalSettings.m_centerFrequency;
|
||
|
}
|
||
|
|
||
|
bool OsmoSDRInput::handleMessage(Message* message)
|
||
|
{
|
||
|
if(MsgConfigureOsmoSDR::match(message)) {
|
||
|
MsgConfigureOsmoSDR* conf = (MsgConfigureOsmoSDR*)message;
|
||
|
if(!applySettings(conf->getGeneralSettings(), conf->getSettings(), false))
|
||
|
qDebug("OsmoSDR config error");
|
||
|
message->completed();
|
||
|
return true;
|
||
|
} else {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
if(cmd->sourceType() != DSPCmdConfigureSourceOsmoSDR::SourceType)
|
||
|
return false;
|
||
|
if(!applySettings(((DSPCmdConfigureSourceOsmoSDR*)cmd)->getSettings(), false))
|
||
|
qDebug("OsmoSDR config error");
|
||
|
cmd->completed();
|
||
|
return true;
|
||
|
*/
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
bool OsmoSDRInput::applySettings(const GeneralSettings& generalSettings, const Settings& settings, bool force)
|
||
|
{
|
||
|
QMutexLocker mutexLocker(&m_mutex);
|
||
|
|
||
|
if((m_generalSettings.m_centerFrequency != generalSettings.m_centerFrequency) || force) {
|
||
|
m_generalSettings.m_centerFrequency = generalSettings.m_centerFrequency;
|
||
|
if(m_dev != NULL) {
|
||
|
if(osmosdr_set_center_freq(m_dev, m_generalSettings.m_centerFrequency) != 0)
|
||
|
qDebug("osmosdr_set_center_freq(%lld) failed", m_generalSettings.m_centerFrequency);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if((m_settings.m_swapIQ != settings.m_swapIQ) || force) {
|
||
|
m_settings.m_swapIQ = settings.m_swapIQ;
|
||
|
if(m_dev != NULL) {
|
||
|
if(osmosdr_set_fpga_iq_swap(m_dev, m_settings.m_swapIQ ? 1 : 0) == 0)
|
||
|
qDebug("osmosdr_set_fpga_iq_swap() failed");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if((m_settings.m_decimation != settings.m_decimation) || force) {
|
||
|
m_settings.m_decimation = settings.m_decimation;
|
||
|
if(m_dev != NULL) {
|
||
|
if(!osmosdr_set_fpga_decimation(m_dev, m_settings.m_decimation))
|
||
|
qDebug("osmosdr_set_fpga_decimation() failed");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if((m_settings.m_lnaGain != settings.m_lnaGain) || force) {
|
||
|
m_settings.m_lnaGain = settings.m_lnaGain;
|
||
|
if(m_dev != NULL) {
|
||
|
if(!osmosdr_set_tuner_lna_gain(m_dev, m_settings.m_lnaGain))
|
||
|
qDebug("osmosdr_set_tuner_lna_gain() failed");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if((m_settings.m_mixerGain != settings.m_mixerGain) || force) {
|
||
|
m_settings.m_mixerGain = settings.m_mixerGain;
|
||
|
if(m_dev != NULL) {
|
||
|
if(!osmosdr_set_tuner_mixer_gain(m_dev, m_settings.m_mixerGain))
|
||
|
qDebug("osmosdr_set_tuner_mixer_gain() failed");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if((m_settings.m_mixerEnhancement != settings.m_mixerEnhancement) || force) {
|
||
|
m_settings.m_mixerEnhancement = settings.m_mixerEnhancement;
|
||
|
if(m_dev != NULL) {
|
||
|
if(!osmosdr_set_tuner_mixer_enh(m_dev, m_settings.m_mixerEnhancement))
|
||
|
qDebug("osmosdr_set_tuner_mixer_enh() failed");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if((m_settings.m_if1gain != settings.m_if1gain) || force) {
|
||
|
m_settings.m_if1gain = settings.m_if1gain;
|
||
|
if(m_dev != NULL) {
|
||
|
if(!osmosdr_set_tuner_if_gain(m_dev, 1, m_settings.m_if1gain))
|
||
|
qDebug("osmosdr_set_tuner_if_gain(1) failed");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if((m_settings.m_if2gain != settings.m_if2gain) || force) {
|
||
|
m_settings.m_if2gain = settings.m_if2gain;
|
||
|
if(m_dev != NULL) {
|
||
|
if(!osmosdr_set_tuner_if_gain(m_dev, 2, m_settings.m_if2gain))
|
||
|
qDebug("osmosdr_set_tuner_if_gain(2) failed");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if((m_settings.m_if3gain != settings.m_if3gain) || force) {
|
||
|
m_settings.m_if3gain = settings.m_if3gain;
|
||
|
if(m_dev != NULL) {
|
||
|
if(!osmosdr_set_tuner_if_gain(m_dev, 3, m_settings.m_if3gain))
|
||
|
qDebug("osmosdr_set_tuner_if_gain(3) failed");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if((m_settings.m_if4gain != settings.m_if4gain) || force) {
|
||
|
m_settings.m_if4gain = settings.m_if4gain;
|
||
|
if(m_dev != NULL) {
|
||
|
if(!osmosdr_set_tuner_if_gain(m_dev, 4, m_settings.m_if4gain))
|
||
|
qDebug("osmosdr_set_tuner_if_gain(4) failed");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if((m_settings.m_if5gain != settings.m_if5gain) || force) {
|
||
|
m_settings.m_if5gain = settings.m_if5gain;
|
||
|
if(m_dev != NULL) {
|
||
|
if(!osmosdr_set_tuner_if_gain(m_dev, 5, m_settings.m_if5gain))
|
||
|
qDebug("osmosdr_set_tuner_if_gain(5) failed");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if((m_settings.m_if6gain != settings.m_if6gain) || force) {
|
||
|
m_settings.m_if6gain = settings.m_if6gain;
|
||
|
if(m_dev != NULL) {
|
||
|
if(!osmosdr_set_tuner_if_gain(m_dev, 6, m_settings.m_if6gain))
|
||
|
qDebug("osmosdr_set_tuner_if_gain(6) failed");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if((m_settings.m_opAmpI1 != settings.m_opAmpI1) || (m_settings.m_opAmpI2 != settings.m_opAmpI2) ||
|
||
|
(m_settings.m_opAmpQ1 != settings.m_opAmpQ1) || (m_settings.m_opAmpQ2 != settings.m_opAmpQ2) ||
|
||
|
force) {
|
||
|
m_settings.m_opAmpI1 = settings.m_opAmpI1;
|
||
|
m_settings.m_opAmpI2 = settings.m_opAmpI2;
|
||
|
m_settings.m_opAmpQ1 = settings.m_opAmpQ1;
|
||
|
m_settings.m_opAmpQ2 = settings.m_opAmpQ2;
|
||
|
if(m_dev != NULL) {
|
||
|
if(!osmosdr_set_iq_amp(m_dev, m_settings.m_opAmpI1, m_settings.m_opAmpI2, m_settings.m_opAmpQ1, m_settings.m_opAmpQ2))
|
||
|
qDebug("osmosdr_set_iq_amp(1) failed");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if((m_settings.m_dcOfsI != settings.m_dcOfsI) || (m_settings.m_dcOfsQ != settings.m_dcOfsQ) ||
|
||
|
force) {
|
||
|
m_settings.m_dcOfsI = settings.m_dcOfsI;
|
||
|
m_settings.m_dcOfsQ = settings.m_dcOfsQ;
|
||
|
if(m_dev != NULL) {
|
||
|
if(!osmosdr_set_tuner_dc_offset(m_dev, m_settings.m_dcOfsI, m_settings.m_dcOfsQ))
|
||
|
qDebug("osmosdr_set_tuner_dc_offset() failed");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|