mirror of
				https://github.com/f4exb/sdrangel.git
				synced 2025-10-25 01:50:21 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			439 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			439 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include <QApplication>
 | |
| #include <QPluginLoader>
 | |
| #include <QComboBox>
 | |
| #include <cstdio>
 | |
| 
 | |
| #include "plugin/pluginmanager.h"
 | |
| #include "plugin/plugingui.h"
 | |
| #include "device/deviceapi.h"
 | |
| #include "settings/preset.h"
 | |
| #include "mainwindow.h"
 | |
| #include "gui/glspectrum.h"
 | |
| #include "dsp/dspdeviceengine.h"
 | |
| #include "util/message.h"
 | |
| 
 | |
| #include <QDebug>
 | |
| 
 | |
| const QString PluginManager::m_sdrDaemonDeviceTypeID = "sdrangel.samplesource.sdrdaemon";
 | |
| const QString PluginManager::m_sdrDaemonFECDeviceTypeID = "sdrangel.samplesource.sdrdaemonfec";
 | |
| const QString PluginManager::m_fileSourceDeviceTypeID = "sdrangel.samplesource.filesource";
 | |
| 
 | |
| PluginManager::PluginManager(MainWindow* mainWindow, QObject* parent) :
 | |
| 	QObject(parent),
 | |
| 	m_pluginAPI(this, mainWindow),
 | |
| 	m_mainWindow(mainWindow)
 | |
| {
 | |
| }
 | |
| 
 | |
| PluginManager::~PluginManager()
 | |
| {
 | |
| //	freeAll();
 | |
| }
 | |
| 
 | |
| void PluginManager::loadPlugins()
 | |
| {
 | |
| 	QString applicationDirPath = QApplication::instance()->applicationDirPath();
 | |
| 	QString applicationLibPath = applicationDirPath + "/../lib";
 | |
| 	qDebug() << "PluginManager::loadPlugins: " << qPrintable(applicationDirPath) << ", " << qPrintable(applicationLibPath);
 | |
| 
 | |
| 	QDir pluginsBinDir = QDir(applicationDirPath);
 | |
| 	QDir pluginsLibDir = QDir(applicationLibPath);
 | |
| 
 | |
| 	loadPlugins(pluginsBinDir);
 | |
| 	loadPlugins(pluginsLibDir);
 | |
| 
 | |
| 	qSort(m_plugins);
 | |
| 
 | |
| 	for (Plugins::const_iterator it = m_plugins.begin(); it != m_plugins.end(); ++it)
 | |
| 	{
 | |
| 		it->pluginInterface->initPlugin(&m_pluginAPI);
 | |
| 	}
 | |
| 
 | |
| 	updateSampleSourceDevices();
 | |
| }
 | |
| 
 | |
| void PluginManager::registerChannel(const QString& channelName, PluginInterface* plugin)
 | |
| {
 | |
| 	m_channelRegistrations.append(PluginAPI::ChannelRegistration(channelName, plugin));
 | |
| }
 | |
| 
 | |
| void PluginManager::registerSampleSource(const QString& sourceName, PluginInterface* plugin)
 | |
| {
 | |
| 	qDebug() << "PluginManager::registerSampleSource "
 | |
| 			<< plugin->getPluginDescriptor().displayedName.toStdString().c_str()
 | |
| 			<< " with source name " << sourceName.toStdString().c_str();
 | |
| 
 | |
| 	m_sampleSourceRegistrations.append(SampleSourceRegistration(sourceName, plugin));
 | |
| }
 | |
| 
 | |
| void PluginManager::updateSampleSourceDevices()
 | |
| {
 | |
| 	m_sampleSourceDevices.clear();
 | |
| 
 | |
| 	for(int i = 0; i < m_sampleSourceRegistrations.count(); ++i)
 | |
| 	{
 | |
| 		PluginInterface::SampleSourceDevices ssd = m_sampleSourceRegistrations[i].m_plugin->enumSampleSources();
 | |
| 
 | |
| 		for(int j = 0; j < ssd.count(); ++j)
 | |
| 		{
 | |
| 			m_sampleSourceDevices.append(SampleSourceDevice(m_sampleSourceRegistrations[i].m_plugin,
 | |
| 					ssd[j].displayedName,
 | |
| 					ssd[j].id,
 | |
| 					ssd[j].serial,
 | |
| 					ssd[j].sequence));
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void PluginManager::duplicateLocalSampleSourceDevices(uint deviceUID)
 | |
| {
 | |
|     if (deviceUID == 0) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     SampleSourceDevice *sdrDaemonSSD0 = 0;
 | |
|     SampleSourceDevice *sdrDaemonFECSSD0 = 0;
 | |
|     SampleSourceDevice *fileSourceSSD0 = 0;
 | |
|     bool duplicateSDRDaemon = true;
 | |
|     bool duplicateSDRDaemonFEC = true;
 | |
|     bool duplicateFileSource = true;
 | |
| 
 | |
|     for(int i = 0; i < m_sampleSourceDevices.count(); ++i)
 | |
|     {
 | |
|         if (m_sampleSourceDevices[i].m_sourceId == m_sdrDaemonDeviceTypeID) // SDRdaemon
 | |
|         {
 | |
|             if (m_sampleSourceDevices[i].m_sourceSequence == 0) { // reference to device 0
 | |
|                 sdrDaemonSSD0 = &m_sampleSourceDevices[i];
 | |
|             }
 | |
|             else if (m_sampleSourceDevices[i].m_sourceSequence == deviceUID) { // already there
 | |
|                 duplicateSDRDaemon = false;
 | |
|             }
 | |
|         }
 | |
|         else if (m_sampleSourceDevices[i].m_sourceId == m_sdrDaemonFECDeviceTypeID) // SDRdaemon with FEC
 | |
|         {
 | |
|             if (m_sampleSourceDevices[i].m_sourceSequence == 0) { // reference to device 0
 | |
|                 sdrDaemonFECSSD0 = &m_sampleSourceDevices[i];
 | |
|             }
 | |
|             else if (m_sampleSourceDevices[i].m_sourceSequence == deviceUID) { // already there
 | |
|                 duplicateSDRDaemonFEC = false;
 | |
|             }
 | |
|         }
 | |
|         else if (m_sampleSourceDevices[i].m_sourceId == m_fileSourceDeviceTypeID) // File Source
 | |
|         {
 | |
|             if (m_sampleSourceDevices[i].m_sourceSequence == 0) { // reference to device 0
 | |
|                 fileSourceSSD0 = &m_sampleSourceDevices[i];
 | |
|             }
 | |
|             else if (m_sampleSourceDevices[i].m_sourceSequence == deviceUID) { // already there
 | |
|                 duplicateFileSource = false;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (sdrDaemonSSD0 && duplicateSDRDaemon) // append item for a new instance
 | |
|     {
 | |
|         m_sampleSourceDevices.append(
 | |
|             SampleSourceDevice(
 | |
|                 sdrDaemonSSD0->m_plugin,
 | |
|                 QString("SDRdaemon[%1]").arg(deviceUID),
 | |
|                 sdrDaemonSSD0->m_sourceId,
 | |
|                 sdrDaemonSSD0->m_sourceSerial,
 | |
|                 deviceUID
 | |
|             )
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     if (sdrDaemonFECSSD0 && duplicateSDRDaemonFEC) // append item for a new instance
 | |
|     {
 | |
|         m_sampleSourceDevices.append(
 | |
|             SampleSourceDevice(
 | |
|                 sdrDaemonFECSSD0->m_plugin,
 | |
|                 QString("SDRdaemonFEC[%1]").arg(deviceUID),
 | |
|                 sdrDaemonFECSSD0->m_sourceId,
 | |
|                 sdrDaemonFECSSD0->m_sourceSerial,
 | |
|                 deviceUID
 | |
|             )
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     if (fileSourceSSD0 && duplicateFileSource) // append item for a new instance
 | |
|     {
 | |
|         m_sampleSourceDevices.append(
 | |
|             SampleSourceDevice(
 | |
|                 fileSourceSSD0->m_plugin,
 | |
|                 QString("FileSource[%1]").arg(deviceUID),
 | |
|                 fileSourceSSD0->m_sourceId,
 | |
|                 fileSourceSSD0->m_sourceSerial,
 | |
|                 deviceUID
 | |
|             )
 | |
|         );
 | |
|     }
 | |
| }
 | |
| 
 | |
| void PluginManager::fillSampleSourceSelector(QComboBox* comboBox, uint deviceUID)
 | |
| {
 | |
| 	comboBox->clear();
 | |
| 
 | |
| 	for(int i = 0; i < m_sampleSourceDevices.count(); i++)
 | |
| 	{
 | |
| 	    // For "local" devices show only ones that concern this device set
 | |
| 	    if ((m_sampleSourceDevices[i].m_sourceId == m_sdrDaemonDeviceTypeID)
 | |
| 	            || (m_sampleSourceDevices[i].m_sourceId == m_sdrDaemonFECDeviceTypeID)
 | |
| 	            || (m_sampleSourceDevices[i].m_sourceId == m_fileSourceDeviceTypeID))
 | |
| 	    {
 | |
| 	        if (deviceUID != m_sampleSourceDevices[i].m_sourceSequence) {
 | |
| 	            continue;
 | |
| 	        }
 | |
| 	    }
 | |
| 
 | |
| 		comboBox->addItem(m_sampleSourceDevices[i].m_displayName, qVariantFromValue((void *) &m_sampleSourceDevices[i]));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int PluginManager::selectSampleSourceByIndex(int index, DeviceAPI *deviceAPI)
 | |
| {
 | |
| 	qDebug("PluginManager::selectSampleSourceByIndex: index: %d", index);
 | |
| 
 | |
| 	if (m_sampleSourceDevices.count() == 0)
 | |
| 	{
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	if (index < 0)
 | |
| 	{
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	if (index >= m_sampleSourceDevices.count())
 | |
| 	{
 | |
| 		index = 0;
 | |
| 	}
 | |
| 
 | |
|     qDebug() << "PluginManager::selectSampleSourceByIndex: m_sampleSource at index " << index
 | |
|             << " id: " << m_sampleSourceDevices[index].m_sourceId.toStdString().c_str()
 | |
|             << " ser: " << m_sampleSourceDevices[index].m_sourceSerial.toStdString().c_str()
 | |
|             << " seq: " << m_sampleSourceDevices[index].m_sourceSequence;
 | |
| 
 | |
|     deviceAPI->stopAcquisition();
 | |
|     deviceAPI->setSampleSourcePluginGUI(0); // this effectively destroys the previous GUI if it exists
 | |
| 
 | |
| 	QWidget *gui;
 | |
| 	PluginGUI *pluginGUI = m_sampleSourceDevices[index].m_plugin->createSampleSourcePluginGUI(m_sampleSourceDevices[index].m_sourceId, &gui, deviceAPI);
 | |
| 
 | |
| 	//	m_sampleSourcePluginGUI = pluginGUI;
 | |
| 	deviceAPI->setSampleSourceSequence(m_sampleSourceDevices[index].m_sourceSequence);
 | |
| 	deviceAPI->setSampleSourceId(m_sampleSourceDevices[index].m_sourceId);
 | |
| 	deviceAPI->setSampleSourceSerial(m_sampleSourceDevices[index].m_sourceSerial);
 | |
| 	deviceAPI->setSampleSourcePluginGUI(pluginGUI);
 | |
| 	deviceAPI->setInputGUI(gui, m_sampleSourceDevices[index].m_displayName);
 | |
| 
 | |
| 	return index;
 | |
| }
 | |
| 
 | |
| int PluginManager::selectFirstSampleSource(const QString& sourceId, DeviceAPI *deviceAPI)
 | |
| {
 | |
| 	qDebug("PluginManager::selectFirstSampleSource by id: [%s]", qPrintable(sourceId));
 | |
| 
 | |
| 	int index = -1;
 | |
| 
 | |
| 	for (int i = 0; i < m_sampleSourceDevices.count(); i++)
 | |
| 	{
 | |
| 		qDebug("*** %s vs %s", qPrintable(m_sampleSourceDevices[i].m_sourceId), qPrintable(sourceId));
 | |
| 
 | |
| 		if(m_sampleSourceDevices[i].m_sourceId == sourceId)
 | |
| 		{
 | |
| 			index = i;
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if(index == -1)
 | |
| 	{
 | |
| 		if(m_sampleSourceDevices.count() > 0)
 | |
| 		{
 | |
| 			index = 0;
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			return -1;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
|     qDebug() << "PluginManager::selectFirstSampleSource: m_sampleSource at index " << index
 | |
|             << " id: " << m_sampleSourceDevices[index].m_sourceId.toStdString().c_str()
 | |
|             << " ser: " << m_sampleSourceDevices[index].m_sourceSerial.toStdString().c_str()
 | |
|             << " seq: " << m_sampleSourceDevices[index].m_sourceSequence;
 | |
| 
 | |
|     deviceAPI->stopAcquisition();
 | |
|     deviceAPI->setSampleSourcePluginGUI(0); // this effectively destroys the previous GUI if it exists
 | |
| 
 | |
|     QWidget *gui;
 | |
| 	PluginGUI *pluginGUI = m_sampleSourceDevices[index].m_plugin->createSampleSourcePluginGUI(m_sampleSourceDevices[index].m_sourceId, &gui, deviceAPI);
 | |
| 
 | |
| 	//	m_sampleSourcePluginGUI = pluginGUI;
 | |
|     deviceAPI->setSampleSourceSequence(m_sampleSourceDevices[index].m_sourceSequence);
 | |
|     deviceAPI->setSampleSourceId(m_sampleSourceDevices[index].m_sourceId);
 | |
|     deviceAPI->setSampleSourceSerial(m_sampleSourceDevices[index].m_sourceSerial);
 | |
|     deviceAPI->setSampleSourcePluginGUI(pluginGUI);
 | |
|     deviceAPI->setInputGUI(gui, m_sampleSourceDevices[index].m_displayName);
 | |
| 
 | |
| 	return index;
 | |
| }
 | |
| 
 | |
| int PluginManager::selectSampleSourceBySerialOrSequence(const QString& sourceId, const QString& sourceSerial, int sourceSequence, DeviceAPI *deviceAPI)
 | |
| {
 | |
| 	qDebug("PluginManager::selectSampleSourceBySequence by sequence: id: %s ser: %s seq: %d", qPrintable(sourceId), qPrintable(sourceSerial), sourceSequence);
 | |
| 
 | |
| 	int index = -1;
 | |
| 	int index_matchingSequence = -1;
 | |
| 	int index_firstOfKind = -1;
 | |
| 
 | |
| 	for (int i = 0; i < m_sampleSourceDevices.count(); i++)
 | |
| 	{
 | |
| 		if (m_sampleSourceDevices[i].m_sourceId == sourceId)
 | |
| 		{
 | |
| 			index_firstOfKind = i;
 | |
| 
 | |
| 			if (m_sampleSourceDevices[i].m_sourceSerial == sourceSerial)
 | |
| 			{
 | |
| 				index = i; // exact match
 | |
| 				break;
 | |
| 			}
 | |
| 
 | |
| 			if (m_sampleSourceDevices[i].m_sourceSequence == sourceSequence)
 | |
| 			{
 | |
| 				index_matchingSequence = i;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if(index == -1) // no exact match
 | |
| 	{
 | |
| 		if (index_matchingSequence == -1) // no matching sequence
 | |
| 		{
 | |
| 			if (index_firstOfKind == -1) // no matching device type
 | |
| 			{
 | |
| 				if(m_sampleSourceDevices.count() > 0) // take first if any
 | |
| 				{
 | |
| 					index = 0;
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					return -1; // return if no device attached
 | |
| 				}
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				index = index_firstOfKind; // take first that matches device type
 | |
| 			}
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			index = index_matchingSequence; // take the one that matches the sequence in the device type
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
|     qDebug() << "PluginManager::selectSampleSourceBySequence: m_sampleSource at index " << index
 | |
|             << " id: " << m_sampleSourceDevices[index].m_sourceId.toStdString().c_str()
 | |
|             << " ser: " << m_sampleSourceDevices[index].m_sourceSerial.toStdString().c_str()
 | |
|             << " seq: " << m_sampleSourceDevices[index].m_sourceSequence;
 | |
| 
 | |
|     deviceAPI->stopAcquisition();
 | |
|     deviceAPI->setSampleSourcePluginGUI(0); // this effectively destroys the previous GUI if it exists
 | |
| 
 | |
|     QWidget *gui;
 | |
| 	PluginGUI *pluginGUI = m_sampleSourceDevices[index].m_plugin->createSampleSourcePluginGUI(m_sampleSourceDevices[index].m_sourceId, &gui, deviceAPI);
 | |
| 
 | |
| 	//	m_sampleSourcePluginGUI = pluginGUI;
 | |
|     deviceAPI->setSampleSourceSequence(m_sampleSourceDevices[index].m_sourceSequence);
 | |
|     deviceAPI->setSampleSourceId(m_sampleSourceDevices[index].m_sourceId);
 | |
|     deviceAPI->setSampleSourceSerial(m_sampleSourceDevices[index].m_sourceSerial);
 | |
|     deviceAPI->setSampleSourcePluginGUI(pluginGUI);
 | |
|     deviceAPI->setInputGUI(gui, m_sampleSourceDevices[index].m_displayName);
 | |
| 
 | |
| 	return index;
 | |
| }
 | |
| 
 | |
| void PluginManager::selectSampleSourceByDevice(void *devicePtr, DeviceAPI *deviceAPI)
 | |
| {
 | |
|     SampleSourceDevice *sampleSourceDevice = (SampleSourceDevice *) devicePtr;
 | |
| 
 | |
|     qDebug() << "PluginManager::selectSampleSourceByDevice: "
 | |
|             << " id: " << sampleSourceDevice->m_sourceId.toStdString().c_str()
 | |
|             << " ser: " << sampleSourceDevice->m_sourceSerial.toStdString().c_str()
 | |
|             << " seq: " << sampleSourceDevice->m_sourceSequence;
 | |
| 
 | |
|     deviceAPI->stopAcquisition();
 | |
|     deviceAPI->setSampleSourcePluginGUI(0); // this effectively destroys the previous GUI if it exists
 | |
| 
 | |
|     QWidget *gui;
 | |
|     PluginGUI *pluginGUI = sampleSourceDevice->m_plugin->createSampleSourcePluginGUI(sampleSourceDevice->m_sourceId, &gui, deviceAPI);
 | |
| 
 | |
|     //  m_sampleSourcePluginGUI = pluginGUI;
 | |
|     deviceAPI->setSampleSourceSequence(sampleSourceDevice->m_sourceSequence);
 | |
|     deviceAPI->setSampleSourceId(sampleSourceDevice->m_sourceId);
 | |
|     deviceAPI->setSampleSourceSerial(sampleSourceDevice->m_sourceSerial);
 | |
|     deviceAPI->setSampleSourcePluginGUI(pluginGUI);
 | |
|     deviceAPI->setInputGUI(gui, sampleSourceDevice->m_displayName);
 | |
| }
 | |
| 
 | |
| void PluginManager::loadPlugins(const QDir& dir)
 | |
| {
 | |
| 	QDir pluginsDir(dir);
 | |
| 
 | |
| 	foreach (QString fileName, pluginsDir.entryList(QDir::Files))
 | |
| 	{
 | |
|         if (fileName.endsWith(".so") || fileName.endsWith(".dll") || fileName.endsWith(".dylib"))
 | |
| 		{
 | |
| 			qDebug() << "PluginManager::loadPlugins: fileName: " << qPrintable(fileName);
 | |
| 
 | |
| 			QPluginLoader* loader = new QPluginLoader(pluginsDir.absoluteFilePath(fileName));
 | |
| 			PluginInterface* plugin = qobject_cast<PluginInterface*>(loader->instance());
 | |
| 
 | |
| 			if (loader->isLoaded())
 | |
| 			{
 | |
| 				qWarning("PluginManager::loadPlugins: loaded plugin %s", qPrintable(fileName));
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				qWarning() << "PluginManager::loadPlugins: " << qPrintable(loader->errorString());
 | |
| 			}
 | |
| 
 | |
| 			if (plugin != 0)
 | |
| 			{
 | |
| 				m_plugins.append(Plugin(fileName, loader, plugin));
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				loader->unload();
 | |
| 			}
 | |
| 
 | |
| 			delete loader; // Valgrind memcheck
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// recursive calls on subdirectories
 | |
| 
 | |
| 	foreach (QString dirName, pluginsDir.entryList(QDir::Dirs | QDir::NoDotAndDotDot))
 | |
| 	{
 | |
| 		loadPlugins(pluginsDir.absoluteFilePath(dirName));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void PluginManager::populateChannelComboBox(QComboBox *channels)
 | |
| {
 | |
|     for(PluginAPI::ChannelRegistrations::iterator it = m_channelRegistrations.begin(); it != m_channelRegistrations.end(); ++it)
 | |
|     {
 | |
|         const PluginDescriptor& pluginDescipror = it->m_plugin->getPluginDescriptor();
 | |
|         channels->addItem(pluginDescipror.displayedName);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void PluginManager::createChannelInstance(int channelPluginIndex, DeviceAPI *deviceAPI)
 | |
| {
 | |
|     if (channelPluginIndex < m_channelRegistrations.size())
 | |
|     {
 | |
|         PluginInterface *pluginInterface = m_channelRegistrations[channelPluginIndex].m_plugin;
 | |
|         pluginInterface->createChannel(m_channelRegistrations[channelPluginIndex].m_channelName, deviceAPI);
 | |
|     }
 | |
| }
 |