2017-04-23 04:33:18 -04:00
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
2019-05-18 00:30:37 -04:00
|
|
|
// Copyright (C) 2016-2019 Edouard Griffiths, F4EXB //
|
2017-04-23 04:33:18 -04:00
|
|
|
// //
|
|
|
|
// 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 //
|
2019-04-11 08:32:15 -04:00
|
|
|
// (at your option) any later version. //
|
2017-04-23 04:33:18 -04:00
|
|
|
// //
|
|
|
|
// 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/>. //
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2017-11-01 17:56:32 -04:00
|
|
|
#include <QCoreApplication>
|
2017-04-23 04:33:18 -04:00
|
|
|
#include <QPluginLoader>
|
2017-10-24 08:15:08 -04:00
|
|
|
#include <QDebug>
|
|
|
|
|
2017-04-23 04:33:18 -04:00
|
|
|
#include <cstdio>
|
2020-04-19 00:13:32 -04:00
|
|
|
#include <algorithm>
|
2017-04-23 04:33:18 -04:00
|
|
|
|
2017-11-01 22:30:54 -04:00
|
|
|
#include "device/deviceenumerator.h"
|
2017-04-23 04:33:18 -04:00
|
|
|
#include "settings/preset.h"
|
|
|
|
#include "util/message.h"
|
|
|
|
#include "dsp/dspdevicesourceengine.h"
|
|
|
|
#include "dsp/dspdevicesinkengine.h"
|
|
|
|
|
2017-10-24 08:15:08 -04:00
|
|
|
#include "plugin/pluginmanager.h"
|
|
|
|
|
2019-05-01 22:02:40 -04:00
|
|
|
const QString PluginManager::m_localInputHardwareID = "LocalInput";
|
|
|
|
const QString PluginManager::m_localInputDeviceTypeID = "sdrangel.samplesource.localinput";
|
2019-02-02 19:52:11 -05:00
|
|
|
const QString PluginManager::m_remoteInputHardwareID = "RemoteInput";
|
|
|
|
const QString PluginManager::m_remoteInputDeviceTypeID = "sdrangel.samplesource.remoteinput";
|
2019-07-07 18:59:04 -04:00
|
|
|
const QString PluginManager::m_fileInputHardwareID = "FileInput";
|
|
|
|
const QString PluginManager::m_fileInputDeviceTypeID = "sdrangel.samplesource.fileinput";
|
2017-11-01 08:03:12 -04:00
|
|
|
|
2019-05-10 17:38:52 -04:00
|
|
|
const QString PluginManager::m_localOutputHardwareID = "LocalOutput";
|
|
|
|
const QString PluginManager::m_localOutputDeviceTypeID = "sdrangel.samplesource.localoutput";
|
2019-02-02 19:52:11 -05:00
|
|
|
const QString PluginManager::m_remoteOutputHardwareID = "RemoteOutput";
|
|
|
|
const QString PluginManager::m_remoteOutputDeviceTypeID = "sdrangel.samplesink.remoteoutput";
|
2020-08-03 18:29:15 -04:00
|
|
|
const QString PluginManager::m_fileOutputHardwareID = "FileOutput";
|
|
|
|
const QString PluginManager::m_fileOutputDeviceTypeID = "sdrangel.samplesink.fileoutput";
|
2017-04-23 04:33:18 -04:00
|
|
|
|
2017-10-24 08:15:08 -04:00
|
|
|
PluginManager::PluginManager(QObject* parent) :
|
2017-04-23 04:33:18 -04:00
|
|
|
QObject(parent),
|
2017-10-24 08:01:31 -04:00
|
|
|
m_pluginAPI(this)
|
2017-04-23 04:33:18 -04:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
PluginManager::~PluginManager()
|
|
|
|
{
|
2019-05-01 09:17:27 -04:00
|
|
|
// freeAll();
|
2017-04-23 04:33:18 -04:00
|
|
|
}
|
|
|
|
|
2017-12-17 04:36:22 -05:00
|
|
|
void PluginManager::loadPlugins(const QString& pluginsSubDir)
|
2017-04-23 04:33:18 -04:00
|
|
|
{
|
2018-08-18 15:30:51 -04:00
|
|
|
loadPluginsPart(pluginsSubDir);
|
|
|
|
loadPluginsFinal();
|
|
|
|
}
|
|
|
|
|
|
|
|
void PluginManager::loadPluginsPart(const QString& pluginsSubDir)
|
|
|
|
{
|
|
|
|
QString applicationDirPath = QCoreApplication::instance()->applicationDirPath();
|
2019-05-01 09:17:27 -04:00
|
|
|
QStringList PluginsPath;
|
|
|
|
|
|
|
|
// NOTE: not the best solution but for now this is
|
|
|
|
// on make install [PREFIX]/bin and [PREFIX]/lib/sdrangel
|
|
|
|
PluginsPath << applicationDirPath + "/../lib/sdrangel/" + pluginsSubDir;
|
|
|
|
// on build
|
|
|
|
PluginsPath << applicationDirPath + "/lib/" + pluginsSubDir;
|
2018-12-27 03:48:30 -05:00
|
|
|
#ifdef __APPLE__
|
2019-05-01 09:17:27 -04:00
|
|
|
// on SDRAngel.app
|
2019-05-03 11:46:01 -04:00
|
|
|
PluginsPath << applicationDirPath + "/../Resources/lib/" + pluginsSubDir;
|
2019-05-24 02:32:40 -04:00
|
|
|
#elif defined(_WIN32) || defined(WIN32)
|
|
|
|
PluginsPath << applicationDirPath + "/" + pluginsSubDir;
|
2018-12-27 03:48:30 -05:00
|
|
|
#endif
|
2017-12-16 07:39:13 -05:00
|
|
|
|
2019-05-01 09:17:27 -04:00
|
|
|
// NOTE: exit on the first folder found
|
|
|
|
bool found = false;
|
|
|
|
foreach (QString dir, PluginsPath)
|
2019-05-18 12:58:35 -04:00
|
|
|
{
|
2019-05-01 09:17:27 -04:00
|
|
|
QDir d(dir);
|
2019-05-16 07:14:38 -04:00
|
|
|
if (d.entryList(QDir::Files).count() == 0) {
|
2019-05-18 12:58:35 -04:00
|
|
|
qDebug("PluginManager::loadPluginsPart folder %s is empty", qPrintable(dir));
|
|
|
|
continue;
|
2019-05-01 09:17:27 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
found = true;
|
|
|
|
loadPluginsDir(d);
|
|
|
|
break;
|
2019-05-18 12:58:35 -04:00
|
|
|
}
|
2019-05-01 09:17:27 -04:00
|
|
|
|
|
|
|
if (!found)
|
2019-05-18 12:58:35 -04:00
|
|
|
{
|
2019-05-01 09:17:27 -04:00
|
|
|
qCritical("No plugins found. Exit immediately.");
|
|
|
|
exit(EXIT_FAILURE);
|
2019-05-18 12:58:35 -04:00
|
|
|
}
|
2018-08-18 15:30:51 -04:00
|
|
|
}
|
2017-12-16 07:39:13 -05:00
|
|
|
|
2018-08-18 15:30:51 -04:00
|
|
|
void PluginManager::loadPluginsFinal()
|
|
|
|
{
|
2020-04-19 00:13:32 -04:00
|
|
|
std::sort(m_plugins.begin(), m_plugins.end());
|
2017-04-23 04:33:18 -04:00
|
|
|
|
2018-08-18 15:30:51 -04:00
|
|
|
for (Plugins::const_iterator it = m_plugins.begin(); it != m_plugins.end(); ++it)
|
|
|
|
{
|
|
|
|
it->pluginInterface->initPlugin(&m_pluginAPI);
|
|
|
|
}
|
2017-04-23 04:33:18 -04:00
|
|
|
|
2018-08-18 15:30:51 -04:00
|
|
|
DeviceEnumerator::instance()->enumerateRxDevices(this);
|
|
|
|
DeviceEnumerator::instance()->enumerateTxDevices(this);
|
2019-05-18 00:30:37 -04:00
|
|
|
DeviceEnumerator::instance()->enumerateMIMODevices(this);
|
2017-04-23 04:33:18 -04:00
|
|
|
}
|
|
|
|
|
2019-12-15 19:03:47 -05:00
|
|
|
void PluginManager::loadPluginsNonDiscoverable(const DeviceUserArgs& deviceUserArgs)
|
|
|
|
{
|
|
|
|
DeviceEnumerator::instance()->addNonDiscoverableDevices(this, deviceUserArgs);
|
|
|
|
}
|
|
|
|
|
2017-11-22 19:19:32 -05:00
|
|
|
void PluginManager::registerRxChannel(const QString& channelIdURI, const QString& channelId, PluginInterface* plugin)
|
2017-04-23 04:33:18 -04:00
|
|
|
{
|
|
|
|
qDebug() << "PluginManager::registerRxChannel "
|
|
|
|
<< plugin->getPluginDescriptor().displayedName.toStdString().c_str()
|
2017-11-22 19:19:32 -05:00
|
|
|
<< " with channel name " << channelIdURI;
|
2017-04-23 04:33:18 -04:00
|
|
|
|
2017-11-22 19:19:32 -05:00
|
|
|
m_rxChannelRegistrations.append(PluginAPI::ChannelRegistration(channelIdURI, channelId, plugin));
|
2017-04-23 04:33:18 -04:00
|
|
|
}
|
|
|
|
|
2017-11-22 19:19:32 -05:00
|
|
|
void PluginManager::registerTxChannel(const QString& channelIdURI, const QString& channelId, PluginInterface* plugin)
|
2017-04-23 04:33:18 -04:00
|
|
|
{
|
|
|
|
qDebug() << "PluginManager::registerTxChannel "
|
|
|
|
<< plugin->getPluginDescriptor().displayedName.toStdString().c_str()
|
2017-11-22 19:19:32 -05:00
|
|
|
<< " with channel name " << channelIdURI;
|
2017-04-23 04:33:18 -04:00
|
|
|
|
2017-11-22 19:19:32 -05:00
|
|
|
m_txChannelRegistrations.append(PluginAPI::ChannelRegistration(channelIdURI, channelId, plugin));
|
2017-04-23 04:33:18 -04:00
|
|
|
}
|
|
|
|
|
2019-09-02 12:36:56 -04:00
|
|
|
void PluginManager::registerMIMOChannel(const QString& channelIdURI, const QString& channelId, PluginInterface* plugin)
|
|
|
|
{
|
|
|
|
qDebug() << "PluginManager::registerMIMOChannel "
|
|
|
|
<< plugin->getPluginDescriptor().displayedName.toStdString().c_str()
|
|
|
|
<< " with channel name " << channelIdURI;
|
|
|
|
|
|
|
|
m_mimoChannelRegistrations.append(PluginAPI::ChannelRegistration(channelIdURI, channelId, plugin));
|
|
|
|
}
|
|
|
|
|
2017-04-23 04:33:18 -04:00
|
|
|
void PluginManager::registerSampleSource(const QString& sourceName, PluginInterface* plugin)
|
|
|
|
{
|
|
|
|
qDebug() << "PluginManager::registerSampleSource "
|
|
|
|
<< plugin->getPluginDescriptor().displayedName.toStdString().c_str()
|
2019-12-15 19:03:47 -05:00
|
|
|
<< " with source name " << sourceName.toStdString().c_str()
|
|
|
|
<< " and hardware id " << plugin->getPluginDescriptor().hardwareId;
|
|
|
|
|
|
|
|
m_sampleSourceRegistrations.append(PluginAPI::SamplingDeviceRegistration(
|
|
|
|
plugin->getPluginDescriptor().hardwareId,
|
|
|
|
sourceName,
|
|
|
|
plugin
|
|
|
|
));
|
2017-04-23 04:33:18 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void PluginManager::registerSampleSink(const QString& sinkName, PluginInterface* plugin)
|
|
|
|
{
|
|
|
|
qDebug() << "PluginManager::registerSampleSink "
|
|
|
|
<< plugin->getPluginDescriptor().displayedName.toStdString().c_str()
|
2019-12-15 19:03:47 -05:00
|
|
|
<< " with sink name " << sinkName.toStdString().c_str()
|
|
|
|
<< " and hardware id " << plugin->getPluginDescriptor().hardwareId;
|
|
|
|
|
|
|
|
m_sampleSinkRegistrations.append(PluginAPI::SamplingDeviceRegistration(
|
|
|
|
plugin->getPluginDescriptor().hardwareId,
|
|
|
|
sinkName,
|
|
|
|
plugin
|
|
|
|
));
|
2017-04-23 04:33:18 -04:00
|
|
|
}
|
|
|
|
|
2019-05-18 00:30:37 -04:00
|
|
|
void PluginManager::registerSampleMIMO(const QString& mimoName, PluginInterface* plugin)
|
|
|
|
{
|
|
|
|
qDebug() << "PluginManager::registerSampleMIMO "
|
|
|
|
<< plugin->getPluginDescriptor().displayedName.toStdString().c_str()
|
2019-12-15 19:03:47 -05:00
|
|
|
<< " with MIMO name " << mimoName.toStdString().c_str()
|
|
|
|
<< " and hardware id " << plugin->getPluginDescriptor().hardwareId;
|
|
|
|
|
|
|
|
m_sampleMIMORegistrations.append(PluginAPI::SamplingDeviceRegistration(
|
|
|
|
plugin->getPluginDescriptor().hardwareId,
|
|
|
|
mimoName,
|
|
|
|
plugin
|
|
|
|
));
|
2019-05-18 00:30:37 -04:00
|
|
|
}
|
|
|
|
|
2020-09-19 19:06:34 -04:00
|
|
|
void PluginManager::registerFeature(const QString& featureIdURI, const QString& featureId, PluginInterface* plugin)
|
2020-09-08 10:47:20 -04:00
|
|
|
{
|
2020-09-19 19:06:34 -04:00
|
|
|
qDebug() << "PluginManager::registerFeature "
|
|
|
|
<< plugin->getPluginDescriptor().displayedName.toStdString().c_str()
|
|
|
|
<< " with channel name " << featureIdURI;
|
2020-09-08 10:47:20 -04:00
|
|
|
|
2020-09-19 19:06:34 -04:00
|
|
|
m_featureRegistrations.append(PluginAPI::FeatureRegistration(featureIdURI, featureId, plugin));
|
2020-09-08 10:47:20 -04:00
|
|
|
}
|
|
|
|
|
2017-12-17 04:36:22 -05:00
|
|
|
void PluginManager::loadPluginsDir(const QDir& dir)
|
2017-04-23 04:33:18 -04:00
|
|
|
{
|
2019-05-18 12:58:35 -04:00
|
|
|
QDir pluginsDir(dir);
|
|
|
|
|
|
|
|
foreach (QString fileName, pluginsDir.entryList(QDir::Files))
|
|
|
|
{
|
|
|
|
if (QLibrary::isLibrary(fileName))
|
|
|
|
{
|
|
|
|
qDebug("PluginManager::loadPluginsDir: fileName: %s", qPrintable(fileName));
|
|
|
|
|
|
|
|
QPluginLoader* pluginLoader = new QPluginLoader(pluginsDir.absoluteFilePath(fileName));
|
|
|
|
if (!pluginLoader->load())
|
|
|
|
{
|
|
|
|
qWarning("PluginManager::loadPluginsDir: %s", qPrintable(pluginLoader->errorString()));
|
|
|
|
delete pluginLoader;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
PluginInterface* instance = qobject_cast<PluginInterface*>(pluginLoader->instance());
|
|
|
|
if (instance == nullptr)
|
|
|
|
{
|
|
|
|
qWarning("PluginManager::loadPluginsDir: Unable to get main instance of plugin: %s", qPrintable(fileName) );
|
|
|
|
delete pluginLoader;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-05-22 02:31:07 -04:00
|
|
|
delete(pluginLoader);
|
|
|
|
|
2019-05-18 12:58:35 -04:00
|
|
|
qInfo("PluginManager::loadPluginsDir: loaded plugin %s", qPrintable(fileName));
|
2019-05-22 02:31:07 -04:00
|
|
|
m_plugins.append(Plugin(fileName, instance));
|
2019-05-18 12:58:35 -04:00
|
|
|
}
|
|
|
|
}
|
2017-04-23 04:33:18 -04:00
|
|
|
}
|
|
|
|
|
2017-11-01 03:32:44 -04:00
|
|
|
void PluginManager::listTxChannels(QList<QString>& list)
|
2017-04-23 04:33:18 -04:00
|
|
|
{
|
2019-05-18 12:58:35 -04:00
|
|
|
list.clear();
|
2017-11-01 03:32:44 -04:00
|
|
|
|
2019-08-01 12:50:21 -04:00
|
|
|
for (PluginAPI::ChannelRegistrations::iterator it = m_txChannelRegistrations.begin(); it != m_txChannelRegistrations.end(); ++it)
|
2017-04-23 04:33:18 -04:00
|
|
|
{
|
2019-05-18 12:58:35 -04:00
|
|
|
const PluginDescriptor& pluginDescipror = it->m_plugin->getPluginDescriptor();
|
|
|
|
list.append(pluginDescipror.displayedName);
|
2017-04-23 04:33:18 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-01 03:32:44 -04:00
|
|
|
void PluginManager::listRxChannels(QList<QString>& list)
|
2017-04-23 04:33:18 -04:00
|
|
|
{
|
2019-05-18 12:58:35 -04:00
|
|
|
list.clear();
|
2017-11-01 03:32:44 -04:00
|
|
|
|
2019-08-01 12:50:21 -04:00
|
|
|
for (PluginAPI::ChannelRegistrations::iterator it = m_rxChannelRegistrations.begin(); it != m_rxChannelRegistrations.end(); ++it)
|
2017-04-23 04:33:18 -04:00
|
|
|
{
|
2019-05-18 12:58:35 -04:00
|
|
|
const PluginDescriptor& pluginDesciptor = it->m_plugin->getPluginDescriptor();
|
|
|
|
list.append(pluginDesciptor.displayedName);
|
2017-04-23 04:33:18 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-02 12:36:56 -04:00
|
|
|
void PluginManager::listMIMOChannels(QList<QString>& list)
|
|
|
|
{
|
|
|
|
list.clear();
|
|
|
|
|
|
|
|
for (PluginAPI::ChannelRegistrations::iterator it = m_mimoChannelRegistrations.begin(); it != m_mimoChannelRegistrations.end(); ++it)
|
|
|
|
{
|
|
|
|
const PluginDescriptor& pluginDesciptor = it->m_plugin->getPluginDescriptor();
|
|
|
|
list.append(pluginDesciptor.displayedName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-19 19:06:34 -04:00
|
|
|
void PluginManager::listFeatures(QList<QString>& list)
|
|
|
|
{
|
|
|
|
list.clear();
|
|
|
|
|
|
|
|
for (PluginAPI::FeatureRegistrations::iterator it = m_featureRegistrations.begin(); it != m_featureRegistrations.end(); ++it)
|
|
|
|
{
|
|
|
|
const PluginDescriptor& pluginDesciptor = it->m_plugin->getPluginDescriptor();
|
|
|
|
list.append(pluginDesciptor.displayedName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-01 12:50:21 -04:00
|
|
|
const PluginInterface *PluginManager::getChannelPluginInterface(const QString& channelIdURI) const
|
|
|
|
{
|
2019-08-03 05:21:46 -04:00
|
|
|
for (PluginAPI::ChannelRegistrations::const_iterator it = m_rxChannelRegistrations.begin(); it != m_rxChannelRegistrations.end(); ++it)
|
2019-08-01 12:50:21 -04:00
|
|
|
{
|
|
|
|
if (it->m_channelIdURI == channelIdURI) {
|
|
|
|
return it->m_plugin;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-03 05:21:46 -04:00
|
|
|
for (PluginAPI::ChannelRegistrations::const_iterator it = m_txChannelRegistrations.begin(); it != m_txChannelRegistrations.end(); ++it)
|
2019-08-01 12:50:21 -04:00
|
|
|
{
|
|
|
|
if (it->m_channelIdURI == channelIdURI) {
|
|
|
|
return it->m_plugin;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
2019-08-03 05:21:46 -04:00
|
|
|
|
|
|
|
const PluginInterface *PluginManager::getDevicePluginInterface(const QString& deviceId) const
|
|
|
|
{
|
|
|
|
for (PluginAPI::SamplingDeviceRegistrations::const_iterator it = m_sampleSourceRegistrations.begin(); it != m_sampleSourceRegistrations.end(); ++it)
|
|
|
|
{
|
|
|
|
if (it->m_deviceId == deviceId) {
|
|
|
|
return it->m_plugin;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (PluginAPI::SamplingDeviceRegistrations::const_iterator it = m_sampleSinkRegistrations.begin(); it != m_sampleSinkRegistrations.end(); ++it)
|
|
|
|
{
|
|
|
|
if (it->m_deviceId == deviceId) {
|
|
|
|
return it->m_plugin;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (PluginAPI::SamplingDeviceRegistrations::const_iterator it = m_sampleMIMORegistrations.begin(); it != m_sampleMIMORegistrations.end(); ++it)
|
|
|
|
{
|
|
|
|
if (it->m_deviceId == deviceId) {
|
|
|
|
return it->m_plugin;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
2020-09-19 19:06:34 -04:00
|
|
|
}
|
2020-09-23 23:38:05 -04:00
|
|
|
|
|
|
|
const PluginInterface *PluginManager::getFeaturePluginInterface(const QString& featureIdURI) const
|
|
|
|
{
|
|
|
|
for (PluginAPI::FeatureRegistrations::const_iterator it = m_featureRegistrations.begin(); it != m_featureRegistrations.end(); ++it)
|
|
|
|
{
|
|
|
|
if (it->m_featureIdURI == featureIdURI) {
|
|
|
|
return it->m_plugin;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|