mirror of
https://github.com/f4exb/sdrangel.git
synced 2024-11-03 15:31:15 -05:00
470 lines
14 KiB
C++
470 lines
14 KiB
C++
///////////////////////////////////////////////////////////////////////////////////
|
|
// Copyright (C) 2016 Edouard Griffiths, F4EXB //
|
|
// //
|
|
// 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 <QDebug>
|
|
|
|
#include "util/simpleserializer.h"
|
|
#include "dsp/dspcommands.h"
|
|
#include "dsp/dspengine.h"
|
|
#include "sdrplaygui.h"
|
|
#include "sdrplayinput.h"
|
|
|
|
#include <device/devicesourceapi.h>
|
|
|
|
#include "sdrplaythread.h"
|
|
|
|
MESSAGE_CLASS_DEFINITION(SDRPlayInput::MsgConfigureSDRPlay, Message)
|
|
MESSAGE_CLASS_DEFINITION(SDRPlayInput::MsgReportSDRPlay, Message)
|
|
|
|
SDRPlayInput::SDRPlayInput(DeviceSourceAPI *deviceAPI) :
|
|
m_deviceAPI(deviceAPI),
|
|
m_settings(),
|
|
m_dev(0),
|
|
m_sdrPlayThread(0),
|
|
m_deviceDescription("SDRPlay")
|
|
{
|
|
}
|
|
|
|
SDRPlayInput::~SDRPlayInput()
|
|
{
|
|
stop();
|
|
}
|
|
|
|
bool SDRPlayInput::init(const Message& cmd)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool SDRPlayInput::start(int device)
|
|
{
|
|
QMutexLocker mutexLocker(&m_mutex);
|
|
|
|
if (m_dev != 0)
|
|
{
|
|
stop();
|
|
}
|
|
|
|
char vendor[256];
|
|
char product[256];
|
|
char serial[256];
|
|
int res;
|
|
int numberOfGains;
|
|
|
|
if (!m_sampleFifo.setSize(96000 * 4))
|
|
{
|
|
qCritical("SDRPlayInput::start: could not allocate SampleFifo");
|
|
return false;
|
|
}
|
|
|
|
if ((res = mirisdr_open(&m_dev, MIRISDR_HW_SDRPLAY, device)) < 0)
|
|
{
|
|
qCritical("SDRPlayInput::start: could not open SDRPlay #%d: %s", device, strerror(errno));
|
|
return false;
|
|
}
|
|
|
|
vendor[0] = '\0';
|
|
product[0] = '\0';
|
|
serial[0] = '\0';
|
|
|
|
if ((res = mirisdr_get_device_usb_strings(device, vendor, product, serial)) < 0)
|
|
{
|
|
qCritical("SDRPlayInput::start: error accessing USB device");
|
|
stop();
|
|
return false;
|
|
}
|
|
|
|
qWarning("SDRPlayInput::start: open: %s %s, SN: %s", vendor, product, serial);
|
|
m_deviceDescription = QString("%1 (SN %2)").arg(product).arg(serial);
|
|
|
|
if ((res = mirisdr_set_sample_rate(m_dev, 2048000)) < 0)
|
|
{
|
|
qCritical("SDRPlayInput::start: could not set sample rate: 2048kS/s");
|
|
stop();
|
|
return false;
|
|
}
|
|
|
|
if ((res = mirisdr_set_center_freq(m_dev, m_settings.m_centerFrequency)) < 0)
|
|
{
|
|
qCritical("SDRPlayInput::start: could not set frequency to: %lu Hz", m_settings.m_centerFrequency);
|
|
stop();
|
|
return false;
|
|
}
|
|
|
|
char s12FormatString[] = "336_S16";
|
|
|
|
if ((res = mirisdr_set_sample_format(m_dev, s12FormatString))) // sample format S12
|
|
{
|
|
qCritical("SDRPlayInput::start: could not set sample format: rc: %d", res);
|
|
stop();
|
|
return false;
|
|
}
|
|
|
|
char bulkFormatString[] = "BULK";
|
|
|
|
if ((res = mirisdr_set_transfer(m_dev, bulkFormatString)) < 0)
|
|
{
|
|
qCritical("SDRPlayInput::start: could not set USB Bulk mode: rc: %d", res);
|
|
stop();
|
|
return false;
|
|
}
|
|
|
|
if ((res = mirisdr_set_if_freq(m_dev, SDRPlayIF::getIF(m_settings.m_ifFrequencyIndex))) < 0)
|
|
{
|
|
qCritical("SDRPlayInput::start: could not set IF frequency at index %d: rc: %d", m_settings.m_ifFrequencyIndex, res);
|
|
stop();
|
|
return false;
|
|
}
|
|
|
|
if ((res = mirisdr_set_bandwidth(m_dev, SDRPlayBandwidths::getBandwidth(m_settings.m_bandwidthIndex))) < 0)
|
|
{
|
|
qCritical("SDRPlayInput::start: could not set bandwidth at index %d: rc: %d", m_settings.m_bandwidthIndex, res);
|
|
stop();
|
|
return false;
|
|
}
|
|
|
|
if ((res = mirisdr_set_tuner_gain_mode(m_dev, 1)) < 0)
|
|
{
|
|
qCritical("SDRPlayInput::start: error setting tuner gain mode");
|
|
stop();
|
|
return false;
|
|
}
|
|
|
|
numberOfGains = mirisdr_get_tuner_gains(m_dev, 0);
|
|
|
|
if (numberOfGains < 0)
|
|
{
|
|
qCritical("SDRPlayInput::start: error getting number of gain values supported");
|
|
stop();
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
qDebug("SDRPlayInput::start: supported gain values: %d", numberOfGains);
|
|
}
|
|
|
|
m_gains.resize(numberOfGains);
|
|
|
|
if (mirisdr_get_tuner_gains(m_dev, &m_gains[0]) < 0)
|
|
{
|
|
qCritical("SDRPlayInput::start: error getting gain values");
|
|
stop();
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
qDebug() << "SDRPlayInput::start: " << m_gains.size() << "gains";
|
|
MsgReportSDRPlay *message = MsgReportSDRPlay::create(m_gains);
|
|
getOutputMessageQueueToGUI()->push(message);
|
|
}
|
|
|
|
if ((res = mirisdr_reset_buffer(m_dev)) < 0)
|
|
{
|
|
qCritical("SDRPlayInput::start: could not reset USB EP buffers: %s", strerror(errno));
|
|
stop();
|
|
return false;
|
|
}
|
|
|
|
if((m_sdrPlayThread = new SDRPlayThread(m_dev, &m_sampleFifo)) == 0)
|
|
{
|
|
qFatal("SDRPlayInput::start: failed to create thread");
|
|
return false;
|
|
}
|
|
|
|
m_sdrPlayThread->startWork();
|
|
|
|
mutexLocker.unlock();
|
|
|
|
applySettings(m_settings, true);
|
|
|
|
return true;
|
|
}
|
|
|
|
void SDRPlayInput::stop()
|
|
{
|
|
QMutexLocker mutexLocker(&m_mutex);
|
|
|
|
if(m_sdrPlayThread != 0)
|
|
{
|
|
m_sdrPlayThread->stopWork();
|
|
delete m_sdrPlayThread;
|
|
m_sdrPlayThread = 0;
|
|
}
|
|
|
|
if (m_dev != 0)
|
|
{
|
|
mirisdr_close(m_dev);
|
|
m_dev = 0;
|
|
}
|
|
|
|
m_deviceDescription.clear();
|
|
}
|
|
|
|
const QString& SDRPlayInput::getDeviceDescription() const
|
|
{
|
|
return m_deviceDescription;
|
|
}
|
|
|
|
int SDRPlayInput::getSampleRate() const
|
|
{
|
|
int rate = SDRPlaySampleRates::getRate(m_settings.m_devSampleRateIndex);
|
|
return rate / (1<<m_settings.m_log2Decim);
|
|
}
|
|
|
|
quint64 SDRPlayInput::getCenterFrequency() const
|
|
{
|
|
return m_settings.m_centerFrequency;
|
|
}
|
|
|
|
bool SDRPlayInput::handleMessage(const Message& message)
|
|
{
|
|
if (MsgConfigureSDRPlay::match(message))
|
|
{
|
|
MsgConfigureSDRPlay& conf = (MsgConfigureSDRPlay&) message;
|
|
qDebug() << "SDRPlayInput::handleMessage: MsgConfigureSDRPlay";
|
|
|
|
if (!applySettings(conf.getSettings(), false))
|
|
{
|
|
qDebug("SDRPlayInput::handleMessage: config error");
|
|
}
|
|
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool SDRPlayInput::applySettings(const SDRPlaySettings& settings, bool force)
|
|
{
|
|
bool forwardChange = false;
|
|
bool forceManualGain = false;
|
|
QMutexLocker mutexLocker(&m_mutex);
|
|
|
|
if ((m_settings.m_dcBlock != settings.m_dcBlock) || force)
|
|
{
|
|
m_settings.m_dcBlock = settings.m_dcBlock;
|
|
m_deviceAPI->configureCorrections(m_settings.m_dcBlock, m_settings.m_iqCorrection);
|
|
}
|
|
|
|
if ((m_settings.m_iqCorrection != settings.m_iqCorrection) || force)
|
|
{
|
|
m_settings.m_iqCorrection = settings.m_iqCorrection;
|
|
m_deviceAPI->configureCorrections(m_settings.m_dcBlock, m_settings.m_iqCorrection);
|
|
}
|
|
|
|
if ((m_settings.m_autoGain != settings.m_autoGain) || force)
|
|
{
|
|
m_settings.m_autoGain = settings.m_autoGain;
|
|
|
|
if(m_dev != 0)
|
|
{
|
|
int r = mirisdr_set_tuner_gain_mode(m_dev, m_settings.m_autoGain ? 0 : 1);
|
|
|
|
if (r < 0)
|
|
{
|
|
qDebug("SDRPlayInput::applySettings: could not set auto gain %s: rc: %d", m_settings.m_autoGain ? "on" : "off", r);
|
|
}
|
|
else
|
|
{
|
|
qDebug("SDRPlayInput::applySettings: auto gain set to %s", m_settings.m_autoGain ? "on" : "off");
|
|
forceManualGain = !m_settings.m_autoGain;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!m_settings.m_autoGain)
|
|
{
|
|
if ((m_settings.m_gain != settings.m_gain) || force || forceManualGain)
|
|
{
|
|
m_settings.m_gain = settings.m_gain;
|
|
|
|
if(m_dev != 0)
|
|
{
|
|
int r = mirisdr_set_tuner_gain(m_dev, m_settings.m_gain);
|
|
|
|
if (r < 0)
|
|
{
|
|
qDebug("SDRPlayInput::applySettings: could not set tuner gain()");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if ((m_settings.m_devSampleRateIndex != settings.m_devSampleRateIndex) || force)
|
|
{
|
|
forwardChange = true;
|
|
|
|
if(m_dev != 0)
|
|
{
|
|
int sampleRate = SDRPlaySampleRates::getRate(m_settings.m_devSampleRateIndex);
|
|
int r = mirisdr_set_sample_rate(m_dev, sampleRate);
|
|
|
|
if(r < 0)
|
|
{
|
|
qCritical("SDRPlayInput::applySettings: could not set sample rate: %d rc: %d", sampleRate, r);
|
|
}
|
|
else
|
|
{
|
|
qDebug("SDRPlayInput::applySettings: sample rate set to %d", sampleRate);
|
|
m_settings.m_devSampleRateIndex = settings.m_devSampleRateIndex;
|
|
m_sdrPlayThread->setSamplerate(sampleRate);
|
|
}
|
|
}
|
|
}
|
|
|
|
if ((m_settings.m_log2Decim != settings.m_log2Decim) || force)
|
|
{
|
|
m_settings.m_log2Decim = settings.m_log2Decim;
|
|
forwardChange = true;
|
|
|
|
if (m_sdrPlayThread != 0)
|
|
{
|
|
m_sdrPlayThread->setLog2Decimation(m_settings.m_log2Decim);
|
|
qDebug() << "SDRPlayInput::applySettings: set decimation to " << (1<<m_settings.m_log2Decim);
|
|
}
|
|
}
|
|
|
|
if (m_settings.m_centerFrequency != settings.m_centerFrequency)
|
|
{
|
|
forwardChange = true;
|
|
}
|
|
|
|
qint64 deviceCenterFrequency = m_settings.m_centerFrequency;
|
|
qint64 f_img = deviceCenterFrequency;
|
|
quint32 devSampleRate = SDRPlaySampleRates::getRate(m_settings.m_devSampleRateIndex);
|
|
|
|
if (force || (m_settings.m_centerFrequency != settings.m_centerFrequency)
|
|
|| (m_settings.m_LOppmTenths != settings.m_LOppmTenths)
|
|
|| (m_settings.m_fcPos != settings.m_fcPos))
|
|
{
|
|
m_settings.m_centerFrequency = settings.m_centerFrequency;
|
|
m_settings.m_LOppmTenths = settings.m_LOppmTenths;
|
|
|
|
if ((m_settings.m_log2Decim == 0) || (settings.m_fcPos == SDRPlaySettings::FC_POS_CENTER))
|
|
{
|
|
deviceCenterFrequency = m_settings.m_centerFrequency;
|
|
f_img = deviceCenterFrequency;
|
|
}
|
|
else
|
|
{
|
|
if (settings.m_fcPos == SDRPlaySettings::FC_POS_INFRA)
|
|
{
|
|
deviceCenterFrequency = m_settings.m_centerFrequency + (devSampleRate / 4);
|
|
f_img = deviceCenterFrequency + devSampleRate/2;
|
|
}
|
|
else if (settings.m_fcPos == SDRPlaySettings::FC_POS_SUPRA)
|
|
{
|
|
deviceCenterFrequency = m_settings.m_centerFrequency - (devSampleRate / 4);
|
|
f_img = deviceCenterFrequency - devSampleRate/2;
|
|
}
|
|
}
|
|
|
|
if(m_dev != 0)
|
|
{
|
|
if (setCenterFrequency(deviceCenterFrequency))
|
|
{
|
|
qDebug() << "SDRPlayInput::applySettings: center freq: " << m_settings.m_centerFrequency << " Hz"
|
|
<< " device center freq: " << deviceCenterFrequency << " Hz"
|
|
<< " device sample rate: " << devSampleRate << "Hz"
|
|
<< " Actual sample rate: " << devSampleRate/(1<<m_settings.m_log2Decim) << "Hz"
|
|
<< " img: " << f_img << "Hz";
|
|
}
|
|
}
|
|
}
|
|
|
|
if ((m_settings.m_fcPos != settings.m_fcPos) || force)
|
|
{
|
|
m_settings.m_fcPos = settings.m_fcPos;
|
|
|
|
if (m_sdrPlayThread != 0)
|
|
{
|
|
m_sdrPlayThread->setFcPos((int) m_settings.m_fcPos);
|
|
qDebug() << "SDRPlayInput: set fc pos (enum) to " << (int) m_settings.m_fcPos;
|
|
}
|
|
}
|
|
|
|
if ((m_settings.m_frequencyBandIndex != settings.m_frequencyBandIndex) || force)
|
|
{
|
|
m_settings.m_frequencyBandIndex = settings.m_frequencyBandIndex;
|
|
// change of frequency is done already
|
|
}
|
|
|
|
if ((m_settings.m_bandwidthIndex != settings.m_bandwidthIndex) || force)
|
|
{
|
|
int bandwidth = SDRPlayBandwidths::getBandwidth(settings.m_bandwidthIndex);
|
|
int r = mirisdr_set_bandwidth(m_dev, bandwidth);
|
|
|
|
if (r < 0)
|
|
{
|
|
qCritical("SDRPlayInput::applySettings: set bandwidth %d failed: rc: %d", bandwidth, r);
|
|
}
|
|
else
|
|
{
|
|
qDebug("SDRPlayInput::applySettings: bandwidth set to %d", bandwidth);
|
|
m_settings.m_bandwidthIndex = settings.m_bandwidthIndex;
|
|
}
|
|
}
|
|
|
|
if (m_settings.m_ifFrequencyIndex != settings.m_ifFrequencyIndex)
|
|
{
|
|
int iFFrequency = SDRPlayIF::getIF(settings.m_ifFrequencyIndex);
|
|
int r = mirisdr_set_if_freq(m_dev, iFFrequency);
|
|
|
|
if (r < 0)
|
|
{
|
|
qCritical("SDRPlayInput::applySettings: set IF frequency to %d failed: rc: %d", iFFrequency, r);
|
|
}
|
|
else
|
|
{
|
|
qDebug("SDRPlayInput::applySettings: IF frequency set to %d", iFFrequency);
|
|
m_settings.m_ifFrequencyIndex = settings.m_ifFrequencyIndex;
|
|
}
|
|
}
|
|
|
|
if (forwardChange)
|
|
{
|
|
int sampleRate = getSampleRate();
|
|
DSPSignalNotification *notif = new DSPSignalNotification(sampleRate, m_settings.m_centerFrequency);
|
|
m_deviceAPI->getDeviceInputMessageQueue()->push(notif);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SDRPlayInput::setCenterFrequency(quint64 freq_hz)
|
|
{
|
|
qint64 df = ((qint64)freq_hz * m_settings.m_LOppmTenths) / 10000000LL;
|
|
freq_hz += df;
|
|
|
|
int r = mirisdr_set_center_freq(m_dev, static_cast<uint32_t>(freq_hz));
|
|
|
|
if (r != 0)
|
|
{
|
|
qWarning("SDRPlayInput::setCenterFrequency: could not frequency to %llu Hz", freq_hz);
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
qWarning("SDRPlayInput::setCenterFrequency: frequency set to %llu Hz", freq_hz);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|