/////////////////////////////////////////////////////////////////////////////////// // 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 . // /////////////////////////////////////////////////////////////////////////////////// #include #include #include #include "util/simpleserializer.h" #include "dsp/dspcommands.h" #include "dsp/dspengine.h" #include "sdrplaygui.h" #include "sdrplayinput.h" #include #include "sdrplaythread.h" MESSAGE_CLASS_DEFINITION(SDRPlayInput::MsgConfigureSDRPlay, Message) MESSAGE_CLASS_DEFINITION(SDRPlayInput::MsgReportSDRPlayGains, Message) SDRPlayInput::SDRPlayInput(DeviceSourceAPI *deviceAPI) : m_deviceAPI(deviceAPI), m_settings(), m_dev(0), m_sdrPlayThread(0), m_deviceDescription("SDRPlay"), m_devNumber(0) { } SDRPlayInput::~SDRPlayInput() { stop(); } bool SDRPlayInput::start(int device) { QMutexLocker mutexLocker(&m_mutex); m_devNumber = device; 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); 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; } int sampleRate = SDRPlaySampleRates::getRate(m_settings.m_devSampleRateIndex); if ((res = mirisdr_set_sample_rate(m_dev, sampleRate))) { qCritical("SDRPlayInput::start: could not set sample rate to %d: rc: %d", sampleRate, 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_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, 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<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); } // gains processing if ((m_settings.m_tunerGainMode != settings.m_tunerGainMode) || force) { m_settings.m_tunerGainMode = settings.m_tunerGainMode; forceGainSetting = true; } if (m_settings.m_tunerGainMode) // auto { if ((m_settings.m_tunerGain != settings.m_tunerGain) || forceGainSetting) { m_settings.m_tunerGain = settings.m_tunerGain; if(m_dev != 0) { int r = mirisdr_set_tuner_gain(m_dev, m_settings.m_tunerGain); if (r < 0) { qDebug("SDRPlayInput::applySettings: could not set tuner gain"); } else { int lnaGain; if (m_settings.m_frequencyBandIndex < 3) // bands using AM mode { lnaGain = mirisdr_get_mixbuffer_gain(m_dev); } else { lnaGain = mirisdr_get_lna_gain(m_dev); } MsgReportSDRPlayGains *message = MsgReportSDRPlayGains::create( lnaGain, mirisdr_get_mixer_gain(m_dev), mirisdr_get_baseband_gain(m_dev), mirisdr_get_tuner_gain(m_dev) ); getOutputMessageQueueToGUI()->push(message); } } } } else // manual { bool anyChange = false; if ((m_settings.m_lnaOn != settings.m_lnaOn) || forceGainSetting) { if(m_dev != 0) { if (m_settings.m_frequencyBandIndex < 3) // bands using AM mode { int r = mirisdr_set_mixbuffer_gain(m_dev, settings.m_lnaOn ? 0 : 1); // mirisdr_set_mixbuffer_gain takes gain reduction if (r != 0) { qDebug("SDRPlayInput::applySettings: could not set mixer buffer gain"); } else { m_settings.m_lnaOn = settings.m_lnaOn; anyChange = true; } } else { int r = mirisdr_set_lna_gain(m_dev, settings.m_lnaOn ? 0 : 1); // mirisdr_set_lna_gain takes gain reduction if (r != 0) { qDebug("SDRPlayInput::applySettings: could not set LNA gain"); } else { m_settings.m_lnaOn = settings.m_lnaOn; anyChange = true; } } } } if ((m_settings.m_mixerAmpOn != settings.m_mixerAmpOn) || forceGainSetting) { if(m_dev != 0) { int r = mirisdr_set_mixer_gain(m_dev, settings.m_mixerAmpOn ? 0 : 1); // mirisdr_set_lna_gain takes gain reduction if (r != 0) { qDebug("SDRPlayInput::applySettings: could not set mixer gain"); } else { m_settings.m_mixerAmpOn = settings.m_mixerAmpOn; anyChange = true; } } } if ((m_settings.m_basebandGain != settings.m_basebandGain) || forceGainSetting) { if(m_dev != 0) { int r = mirisdr_set_baseband_gain(m_dev, settings.m_basebandGain); if (r != 0) { qDebug("SDRPlayInput::applySettings: could not set mixer gain"); } else { m_settings.m_basebandGain = settings.m_basebandGain; anyChange = true; } } } if (anyChange) { int lnaGain; if (m_settings.m_frequencyBandIndex < 3) // bands using AM mode { lnaGain = mirisdr_get_mixbuffer_gain(m_dev); } else { mirisdr_get_lna_gain(m_dev); } MsgReportSDRPlayGains *message = MsgReportSDRPlayGains::create( lnaGain, mirisdr_get_mixer_gain(m_dev), mirisdr_get_baseband_gain(m_dev), mirisdr_get_tuner_gain(m_dev) ); getOutputMessageQueueToGUI()->push(message); } } 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<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) || force) { 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(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; } }