mirror of
https://github.com/f4exb/sdrangel.git
synced 2024-11-30 03:38:55 -05:00
411 lines
17 KiB
C++
411 lines
17 KiB
C++
///////////////////////////////////////////////////////////////////////////////////
|
|
// Copyright (C) 2017 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 //
|
|
// (at your option) any later version. //
|
|
// //
|
|
// 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 "dsp/dspdevicesourceengine.h"
|
|
#include "dsp/dspdevicesinkengine.h"
|
|
#include "plugin/pluginapi.h"
|
|
#include "plugin/plugininterface.h"
|
|
#include "settings/preset.h"
|
|
#include "channel/channelapi.h"
|
|
#include "settings/preset.h"
|
|
|
|
#include "deviceset.h"
|
|
|
|
|
|
DeviceSet::ChannelInstanceRegistration::ChannelInstanceRegistration(const QString& channelName, ChannelAPI* channelAPI) :
|
|
m_channelName(channelName),
|
|
m_channelSinkAPI(channelAPI),
|
|
m_channelSourceAPI(0)
|
|
{
|
|
if (channelAPI->getStreamType() == ChannelAPI::StreamSingleSink)
|
|
{
|
|
m_channelSinkAPI = channelAPI;
|
|
m_channelSourceAPI = nullptr;
|
|
}
|
|
else if (channelAPI->getStreamType() == ChannelAPI::StreamSingleSource)
|
|
{
|
|
m_channelSinkAPI = nullptr;
|
|
m_channelSourceAPI = channelAPI;
|
|
}
|
|
else
|
|
{
|
|
m_channelSinkAPI = nullptr;
|
|
m_channelSourceAPI = nullptr;
|
|
}
|
|
}
|
|
|
|
DeviceSet::DeviceSet(int tabIndex)
|
|
{
|
|
m_deviceAPI = nullptr;
|
|
m_deviceSourceEngine = nullptr;
|
|
m_deviceSinkEngine = nullptr;
|
|
m_deviceMIMOEngine = nullptr;
|
|
m_deviceTabIndex = tabIndex;
|
|
}
|
|
|
|
DeviceSet::~DeviceSet()
|
|
{
|
|
}
|
|
|
|
void DeviceSet::registerRxChannelInstance(const QString& channelName, ChannelAPI* channelAPI)
|
|
{
|
|
m_rxChannelInstanceRegistrations.append(ChannelInstanceRegistration(channelName, channelAPI));
|
|
renameRxChannelInstances();
|
|
}
|
|
|
|
void DeviceSet::registerTxChannelInstance(const QString& channelName, ChannelAPI* channelAPI)
|
|
{
|
|
m_txChannelInstanceRegistrations.append(ChannelInstanceRegistration(channelName, channelAPI));
|
|
renameTxChannelInstances();
|
|
}
|
|
|
|
void DeviceSet::removeRxChannelInstance(ChannelAPI* channelAPI)
|
|
{
|
|
for(ChannelInstanceRegistrations::iterator it = m_rxChannelInstanceRegistrations.begin(); it != m_rxChannelInstanceRegistrations.end(); ++it)
|
|
{
|
|
if(it->m_channelSinkAPI == channelAPI)
|
|
{
|
|
m_rxChannelInstanceRegistrations.erase(it);
|
|
break;
|
|
}
|
|
}
|
|
|
|
renameRxChannelInstances();
|
|
}
|
|
|
|
void DeviceSet::removeTxChannelInstance(ChannelAPI* channelAPI)
|
|
{
|
|
for(ChannelInstanceRegistrations::iterator it = m_txChannelInstanceRegistrations.begin(); it != m_txChannelInstanceRegistrations.end(); ++it)
|
|
{
|
|
if(it->m_channelSourceAPI == channelAPI)
|
|
{
|
|
m_txChannelInstanceRegistrations.erase(it);
|
|
break;
|
|
}
|
|
}
|
|
|
|
renameTxChannelInstances();
|
|
}
|
|
|
|
void DeviceSet::freeRxChannels()
|
|
{
|
|
for(int i = 0; i < m_rxChannelInstanceRegistrations.count(); i++)
|
|
{
|
|
qDebug("DeviceSet::freeAll: destroying channel [%s]", qPrintable(m_rxChannelInstanceRegistrations[i].m_channelName));
|
|
m_rxChannelInstanceRegistrations[i].m_channelSinkAPI->destroy();
|
|
}
|
|
}
|
|
|
|
void DeviceSet::freeTxChannels()
|
|
{
|
|
for(int i = 0; i < m_txChannelInstanceRegistrations.count(); i++)
|
|
{
|
|
qDebug("DeviceSet::freeAll: destroying channel [%s]", qPrintable(m_txChannelInstanceRegistrations[i].m_channelName));
|
|
m_txChannelInstanceRegistrations[i].m_channelSourceAPI->destroy();
|
|
}
|
|
}
|
|
|
|
void DeviceSet::deleteRxChannel(int channelIndex)
|
|
{
|
|
if (channelIndex < m_rxChannelInstanceRegistrations.count())
|
|
{
|
|
m_rxChannelInstanceRegistrations[channelIndex].m_channelSinkAPI->destroy();
|
|
m_rxChannelInstanceRegistrations.removeAt(channelIndex);
|
|
renameRxChannelInstances();
|
|
}
|
|
}
|
|
|
|
void DeviceSet::deleteTxChannel(int channelIndex)
|
|
{
|
|
if (channelIndex < m_txChannelInstanceRegistrations.count())
|
|
{
|
|
m_txChannelInstanceRegistrations[channelIndex].m_channelSourceAPI->destroy();
|
|
m_txChannelInstanceRegistrations.removeAt(channelIndex);
|
|
renameTxChannelInstances();
|
|
}
|
|
}
|
|
|
|
void DeviceSet::addRxChannel(int selectedChannelIndex, PluginAPI *pluginAPI)
|
|
{
|
|
PluginAPI::ChannelRegistrations *channelRegistrations = pluginAPI->getRxChannelRegistrations(); // Available channel plugins
|
|
ChannelAPI *rxChannel =(*channelRegistrations)[selectedChannelIndex].m_plugin->createRxChannelCS(m_deviceAPI);
|
|
ChannelInstanceRegistration reg = ChannelInstanceRegistration(rxChannel->getName(), rxChannel);
|
|
m_rxChannelInstanceRegistrations.append(reg);
|
|
qDebug("DeviceSet::addRxChannel: %s", qPrintable(rxChannel->getName()));
|
|
}
|
|
|
|
void DeviceSet::addTxChannel(int selectedChannelIndex, PluginAPI *pluginAPI)
|
|
{
|
|
PluginAPI::ChannelRegistrations *channelRegistrations = pluginAPI->getTxChannelRegistrations(); // Available channel plugins
|
|
ChannelAPI *txChannel = (*channelRegistrations)[selectedChannelIndex].m_plugin->createTxChannelCS(m_deviceAPI);
|
|
ChannelInstanceRegistration reg = ChannelInstanceRegistration(txChannel->getName(), txChannel);
|
|
m_txChannelInstanceRegistrations.append(reg);
|
|
qDebug("DeviceSet::addTxChannel: %s", qPrintable(txChannel->getName()));
|
|
}
|
|
|
|
void DeviceSet::loadRxChannelSettings(const Preset *preset, PluginAPI *pluginAPI)
|
|
{
|
|
if (preset->isSourcePreset())
|
|
{
|
|
qDebug("DeviceSet::loadChannelSettings: Loading preset [%s | %s]", qPrintable(preset->getGroup()), qPrintable(preset->getDescription()));
|
|
|
|
// Available channel plugins
|
|
PluginAPI::ChannelRegistrations *channelRegistrations = pluginAPI->getRxChannelRegistrations();
|
|
|
|
// copy currently open channels and clear list
|
|
ChannelInstanceRegistrations openChannels = m_rxChannelInstanceRegistrations;
|
|
m_rxChannelInstanceRegistrations.clear();
|
|
|
|
qDebug("DeviceSet::loadChannelSettings: %d channel(s) in preset", preset->getChannelCount());
|
|
|
|
for (int i = 0; i < preset->getChannelCount(); i++)
|
|
{
|
|
const Preset::ChannelConfig& channelConfig = preset->getChannelConfig(i);
|
|
ChannelInstanceRegistration reg;
|
|
|
|
// if we have one instance available already, use it
|
|
|
|
for (int i = 0; i < openChannels.count(); i++)
|
|
{
|
|
qDebug("DeviceSet::loadChannelSettings: channels compare [%s] vs [%s]", qPrintable(openChannels[i].m_channelName), qPrintable(channelConfig.m_channelIdURI));
|
|
|
|
//if(openChannels[i].m_channelName == channelConfig.m_channelIdURI)
|
|
if (compareRxChannelURIs(openChannels[i].m_channelName, channelConfig.m_channelIdURI))
|
|
{
|
|
qDebug("DeviceSet::loadChannelSettings: channel [%s] found", qPrintable(openChannels[i].m_channelName));
|
|
reg = openChannels.takeAt(i);
|
|
m_rxChannelInstanceRegistrations.append(reg);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// if we haven't one already, create one
|
|
|
|
if (reg.m_channelSinkAPI == 0)
|
|
{
|
|
for (int i = 0; i < channelRegistrations->count(); i++)
|
|
{
|
|
//if((*channelRegistrations)[i].m_channelIdURI == channelConfig.m_channelIdURI)
|
|
if (compareRxChannelURIs((*channelRegistrations)[i].m_channelIdURI, channelConfig.m_channelIdURI))
|
|
{
|
|
qDebug("DeviceSet::loadChannelSettings: creating new channel [%s] from config [%s]",
|
|
qPrintable((*channelRegistrations)[i].m_channelIdURI),
|
|
qPrintable(channelConfig.m_channelIdURI));
|
|
ChannelAPI *rxChannel = (*channelRegistrations)[i].m_plugin->createRxChannelCS(m_deviceAPI);
|
|
reg = ChannelInstanceRegistration(channelConfig.m_channelIdURI, rxChannel);
|
|
m_rxChannelInstanceRegistrations.append(reg);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (reg.m_channelSinkAPI != 0)
|
|
{
|
|
qDebug("DeviceSet::loadChannelSettings: deserializing channel [%s]", qPrintable(channelConfig.m_channelIdURI));
|
|
reg.m_channelSinkAPI->deserialize(channelConfig.m_config);
|
|
}
|
|
}
|
|
|
|
// everything, that is still "available" is not needed anymore
|
|
for (int i = 0; i < openChannels.count(); i++)
|
|
{
|
|
qDebug("DeviceSet::loadChannelSettings: destroying spare channel [%s]", qPrintable(openChannels[i].m_channelName));
|
|
openChannels[i].m_channelSinkAPI->destroy();
|
|
}
|
|
|
|
renameRxChannelInstances();
|
|
}
|
|
else
|
|
{
|
|
qDebug("DeviceSet::loadChannelSettings: Loading preset [%s | %s] not a source preset", qPrintable(preset->getGroup()), qPrintable(preset->getDescription()));
|
|
}
|
|
}
|
|
|
|
void DeviceSet::saveRxChannelSettings(Preset *preset)
|
|
{
|
|
if (preset->isSourcePreset())
|
|
{
|
|
qSort(m_rxChannelInstanceRegistrations.begin(), m_rxChannelInstanceRegistrations.end()); // sort by increasing delta frequency and type
|
|
|
|
for(int i = 0; i < m_rxChannelInstanceRegistrations.count(); i++)
|
|
{
|
|
qDebug("DeviceSet::saveChannelSettings: channel [%s] saved", qPrintable(m_rxChannelInstanceRegistrations[i].m_channelName));
|
|
preset->addChannel(m_rxChannelInstanceRegistrations[i].m_channelName, m_rxChannelInstanceRegistrations[i].m_channelSinkAPI->serialize());
|
|
}
|
|
}
|
|
else
|
|
{
|
|
qDebug("DeviceSet::saveChannelSettings: not a source preset");
|
|
}
|
|
}
|
|
|
|
void DeviceSet::loadTxChannelSettings(const Preset *preset, PluginAPI *pluginAPI)
|
|
{
|
|
if (preset->isSourcePreset())
|
|
{
|
|
qDebug("DeviceSet::loadChannelSettings: Loading preset [%s | %s] not a sink preset", qPrintable(preset->getGroup()), qPrintable(preset->getDescription()));
|
|
}
|
|
else
|
|
{
|
|
qDebug("DeviceSet::loadChannelSettings: Loading preset [%s | %s]", qPrintable(preset->getGroup()), qPrintable(preset->getDescription()));
|
|
|
|
// Available channel plugins
|
|
PluginAPI::ChannelRegistrations *channelRegistrations = pluginAPI->getTxChannelRegistrations();
|
|
|
|
// copy currently open channels and clear list
|
|
ChannelInstanceRegistrations openChannels = m_txChannelInstanceRegistrations;
|
|
m_txChannelInstanceRegistrations.clear();
|
|
|
|
qDebug("DeviceSet::loadChannelSettings: %d channel(s) in preset", preset->getChannelCount());
|
|
|
|
for(int i = 0; i < preset->getChannelCount(); i++)
|
|
{
|
|
const Preset::ChannelConfig& channelConfig = preset->getChannelConfig(i);
|
|
ChannelInstanceRegistration reg;
|
|
|
|
// if we have one instance available already, use it
|
|
|
|
for(int i = 0; i < openChannels.count(); i++)
|
|
{
|
|
qDebug("DeviceSet::loadChannelSettings: channels compare [%s] vs [%s]", qPrintable(openChannels[i].m_channelName), qPrintable(channelConfig.m_channelIdURI));
|
|
|
|
if(openChannels[i].m_channelName == channelConfig.m_channelIdURI)
|
|
{
|
|
qDebug("DeviceSet::loadChannelSettings: channel [%s] found", qPrintable(openChannels[i].m_channelName));
|
|
reg = openChannels.takeAt(i);
|
|
m_txChannelInstanceRegistrations.append(reg);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// if we haven't one already, create one
|
|
|
|
if(reg.m_channelSourceAPI == 0)
|
|
{
|
|
for(int i = 0; i < channelRegistrations->count(); i++)
|
|
{
|
|
if((*channelRegistrations)[i].m_channelIdURI == channelConfig.m_channelIdURI)
|
|
{
|
|
qDebug("DeviceSet::loadChannelSettings: creating new channel [%s]", qPrintable(channelConfig.m_channelIdURI));
|
|
ChannelAPI *txChannel = (*channelRegistrations)[i].m_plugin->createTxChannelCS(m_deviceAPI);
|
|
reg = ChannelInstanceRegistration(channelConfig.m_channelIdURI, txChannel);
|
|
m_txChannelInstanceRegistrations.append(reg);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(reg.m_channelSourceAPI != 0)
|
|
{
|
|
qDebug("DeviceSet::loadChannelSettings: deserializing channel [%s]", qPrintable(channelConfig.m_channelIdURI));
|
|
reg.m_channelSourceAPI->deserialize(channelConfig.m_config);
|
|
}
|
|
}
|
|
|
|
// everything, that is still "available" is not needed anymore
|
|
for(int i = 0; i < openChannels.count(); i++)
|
|
{
|
|
qDebug("DeviceSet::loadChannelSettings: destroying spare channel [%s]", qPrintable(openChannels[i].m_channelName));
|
|
openChannels[i].m_channelSourceAPI->destroy();
|
|
}
|
|
|
|
renameTxChannelInstances();
|
|
}
|
|
}
|
|
|
|
void DeviceSet::saveTxChannelSettings(Preset *preset)
|
|
{
|
|
if (preset->isSourcePreset())
|
|
{
|
|
qDebug("DeviceSet::saveChannelSettings: not a sink preset");
|
|
}
|
|
else
|
|
{
|
|
qSort(m_txChannelInstanceRegistrations.begin(), m_txChannelInstanceRegistrations.end()); // sort by increasing delta frequency and type
|
|
|
|
for(int i = 0; i < m_txChannelInstanceRegistrations.count(); i++)
|
|
{
|
|
qDebug("DeviceSet::saveChannelSettings: channel [%s] saved", qPrintable(m_txChannelInstanceRegistrations[i].m_channelName));
|
|
preset->addChannel(m_txChannelInstanceRegistrations[i].m_channelName, m_txChannelInstanceRegistrations[i].m_channelSourceAPI->serialize());
|
|
}
|
|
}
|
|
}
|
|
|
|
void DeviceSet::renameRxChannelInstances()
|
|
{
|
|
for(int i = 0; i < m_rxChannelInstanceRegistrations.count(); i++)
|
|
{
|
|
m_rxChannelInstanceRegistrations[i].m_channelSinkAPI->setName(QString("%1:%2").arg(m_rxChannelInstanceRegistrations[i].m_channelName).arg(i));
|
|
}
|
|
}
|
|
|
|
void DeviceSet::renameTxChannelInstances()
|
|
{
|
|
for(int i = 0; i < m_txChannelInstanceRegistrations.count(); i++)
|
|
{
|
|
m_txChannelInstanceRegistrations[i].m_channelSourceAPI->setName(QString("%1:%2").arg(m_txChannelInstanceRegistrations[i].m_channelName).arg(i));
|
|
}
|
|
}
|
|
|
|
// sort by increasing delta frequency and type (i.e. name)
|
|
bool DeviceSet::ChannelInstanceRegistration::operator<(const ChannelInstanceRegistration& other) const
|
|
{
|
|
if (m_channelSinkAPI && other.m_channelSinkAPI)
|
|
{
|
|
if (m_channelSinkAPI->getCenterFrequency() == other.m_channelSinkAPI->getCenterFrequency())
|
|
{
|
|
return m_channelSinkAPI->getName() < other.m_channelSinkAPI->getName();
|
|
}
|
|
else
|
|
{
|
|
return m_channelSinkAPI->getCenterFrequency() < other.m_channelSinkAPI->getCenterFrequency();
|
|
}
|
|
}
|
|
else if (m_channelSourceAPI && other.m_channelSourceAPI)
|
|
{
|
|
if (m_channelSourceAPI->getCenterFrequency() == other.m_channelSourceAPI->getCenterFrequency())
|
|
{
|
|
return m_channelSourceAPI->getName() < other.m_channelSourceAPI->getName();
|
|
}
|
|
else
|
|
{
|
|
return m_channelSourceAPI->getCenterFrequency() < other.m_channelSourceAPI->getCenterFrequency();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool DeviceSet::compareRxChannelURIs(const QString& registerdChannelURI, const QString& xChannelURI)
|
|
{
|
|
if ((xChannelURI == "sdrangel.channel.chanalyzerng") || (xChannelURI == "sdrangel.channel.chanalyzer")) { // renamed ChanalyzerNG to Chanalyzer in 4.0.0
|
|
return registerdChannelURI == "sdrangel.channel.chanalyzer";
|
|
} else if ((xChannelURI == "de.maintech.sdrangelove.channel.am") || (xChannelURI == "sdrangel.channel.amdemod")) {
|
|
return registerdChannelURI == "sdrangel.channel.amdemod";
|
|
} else if ((xChannelURI == "de.maintech.sdrangelove.channel.nfm") || (xChannelURI == "sdrangel.channel.nfmdemod")) {
|
|
return registerdChannelURI == "sdrangel.channel.nfmdemod";
|
|
} else if ((xChannelURI == "de.maintech.sdrangelove.channel.ssb") || (xChannelURI == "sdrangel.channel.ssbdemod")) {
|
|
return registerdChannelURI == "sdrangel.channel.ssbdemod";
|
|
} else if ((xChannelURI == "de.maintech.sdrangelove.channel.wfm") || (xChannelURI == "sdrangel.channel.wfmdemod")) {
|
|
return registerdChannelURI == "sdrangel.channel.wfmdemod";
|
|
} else {
|
|
return registerdChannelURI == xChannelURI;
|
|
}
|
|
}
|