///////////////////////////////////////////////////////////////////////////////////
// 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 . //
///////////////////////////////////////////////////////////////////////////////////
#include
#include
#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;
}