plugins_manager.cpp 7.59 KB
Newer Older
1
#include "plugins_manager.h"
2 3
#include "rational_function.h"
#include "vertical_segment.h"
4 5

#include <QCoreApplication>
6
/*
7 8
#include <QPluginLoader>
#include <QtPlugin>
9
#include <QLibrary>
10
*/
11 12
#include <QDir>

13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45

#ifdef WIN32
#else
    #include <stdio.h>
    #include <dlfcn.h>
#endif

//! \brief Open a dynamic library file (.so or .dll) and extract the associated
//! provide function. The template argument is used to cast the library to a
//! specific type.
template<typename T> T open_library(const std::string& filename, const char* function)
{
#ifdef WIN32
    return NULL;
#else
    void* handle = dlopen(filename.c_str(), RTLD_LAZY);
    if(handle != NULL)
    {
        void* res = dlsym(handle, function);

        if(dlerror() != NULL)
        {
            std::cerr << "<<ERROR>> unable to load the symbol \"" << function << "\" from " << filename << std::endl;
            return NULL;
        }
#ifdef DEBUG_CORE
        std::cout << "<<DEBUG>> will provide a " << function << " for library \"" << filename << "\"" << std::endl;
#endif
        return (T)res;
    }
    else
    {
        std::cerr << "<<ERROR>> unable to load the dynamic library file \"" << filename << "\"" << std::endl;
46
		  std::cerr << "          cause: \"" << dlerror() << "\"" << std::endl;
47 48 49 50 51 52 53
        return NULL;
    }
#endif
}


// Create the object, parse the argument and load all the plugins
54 55 56 57 58 59 60 61 62 63 64 65 66
plugins_manager::plugins_manager(const arguments& args) 
{

	QDir pluginsDir;
	if(args.is_defined("plugins"))
	{
		pluginsDir = QDir(args["plugins"].c_str()) ;
	}
	else if(QCoreApplication::instance() != NULL)
	{
		pluginsDir = QDir(QCoreApplication::instance()->applicationDirPath());
	}

67
	/*
68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120
	foreach (QString fileName, pluginsDir.entryList(QDir::Files)) 
	{
		QPluginLoader loader ;
		loader.setLoadHints(QLibrary::ExportExternalSymbolsHint) ;
		loader.setFileName(pluginsDir.absoluteFilePath(fileName));
		
		// Convert filename for outputing		
		std::string filename(fileName.toStdString()) ;

		if(!loader.load())
		{
#ifdef DEBUG
			std::cout << "<<DEBUG>> " << loader.errorString().toStdString() << std::endl ;
#endif
			continue ;
		}

		QObject *plugin = loader.instance();
		if (plugin != NULL) 
		{
#ifdef DEBUG
			std::cout << "<<DEBUG>> loading plugin " << filename << std::endl ;
#endif
			if(dynamic_cast<function*>(plugin) != NULL)
			{
#ifdef DEBUG
				std::cout << "<<DEBUG>>  -> it is a function" << std::endl ;
#endif
				_functions.insert(std::pair<std::string, function*>(filename, dynamic_cast<function*>(plugin))) ;
			}
			
			if(dynamic_cast<data*>(plugin) != NULL)
			{
#ifdef DEBUG
				std::cout << "<<DEBUG>>  -> it is a data loader" << std::endl ;
#endif
				_datas.insert(std::pair<std::string, data*>(filename, dynamic_cast<data*>(plugin))) ;
			}
			if(dynamic_cast<fitter*>(plugin) != NULL)
			{
#ifdef DEBUG
				std::cout << "<<DEBUG>>  -> it is a fitter" << std::endl ;
#endif
				_fitters.insert(std::pair<std::string, fitter*>(filename, dynamic_cast<fitter*>(plugin))) ;
			}
		}
		else
		{
#ifdef DEBUG
			std::cout << "<<DEBUG>> " << loader.errorString().toStdString() << std::endl ;
#endif
		}
	}
121
	*/
122 123 124 125 126
}

// Get instances of the function, the data and the
// fitter
//
127
function* plugins_manager::get_function()
128
{
129 130 131 132
#ifdef USING_STATIC
    return new rational_function() ;
#else
    if(_functions.empty())
133
	{
134
		return new rational_function() ;
135 136 137 138 139
	}
	else
	{
		return _functions.begin()->second ;
	}
140 141
#endif

142
}
143
data* plugins_manager::get_data()
144
{
Laurent Belcour's avatar
Laurent Belcour committed
145
	//if(_datas.empty())
146
	{
147 148 149
#ifdef DEBUG
        std::cout << "<<DEBUG>>  using vertical segment data loader" << std::endl ;
#endif
150
		return new vertical_segment() ;
Laurent Belcour's avatar
Laurent Belcour committed
151
	}/*
152 153
	else
	{
154
#ifdef DEBUG
155
		std::cout << "<<DEBUG>>  using \"" << _datas.begin()->first << "\" data loader" << std::endl ;
156
#endif
157
		return _datas.begin()->second ;
Laurent Belcour's avatar
Laurent Belcour committed
158
	}*/
159
}
160
fitter* plugins_manager::get_fitter()
161
{
162 163 164
#ifdef USING_STATIC
    return NULL;
#else
165 166 167 168 169 170
	if(_fitters.empty())
	{
		return NULL ;
	}
	else
	{
171
#ifdef DEBUG
172
		std::cout << "<<DEBUG>>  using \"" <<  _fitters.begin()->first << "\"" << std::endl ;
173
#endif
174 175
		return _fitters.begin()->second ;
	}
176
#endif
177 178 179 180 181 182
}

// Get instances of the function, the data and the
// fitter, select one based on the name. Return null
// if no one exist.
//
183
function* plugins_manager::get_function(const std::string& n)
184
{
185 186 187 188 189 190 191 192
    if(n.empty())
    {
#ifdef DEBUG
        std::cout << "<<DEBUG>> no function plugin specified, returning a rational function" << std::endl;
#endif
        return new rational_function();
    }

193
#ifdef USING_STATIC
Laurent Belcour's avatar
Laurent Belcour committed
194 195 196 197 198 199 200 201 202 203 204
	 std::string file;
	 if(n[0] == '.')
	 {
		 file = n.substr(1, n.size()-1);
	 }
	 else
	 {
		 file = n;
	 }

    QString path = QDir::currentPath() + QString(file.c_str()) ;
205 206
	 
	 FunctionPrototype myFunction = open_library<FunctionPrototype>(path.toStdString(), "provide_function");
207 208 209 210 211 212 213 214 215 216 217 218 219 220
    if(myFunction != NULL)
    {
#ifdef DEBUG
        std::cout << "<<DEBUG>> using function provider in file \"" << n << "\"" << std::endl;
#endif
        return myFunction();
    }
    else
    {
        std::cerr << "<<ERROR>> no function provider found in file \"" << n << "\"" << std::endl;
        return new rational_function() ;
    }
#else

221
	if(_functions.count(n) == 0)
222
	{
223
		return new rational_function() ;
224 225 226
	}
	else
	{
Laurent Belcour's avatar
Laurent Belcour committed
227
		return _functions.find(n)->second ;
228
	}
229
#endif
230
}
231
data* plugins_manager::get_data(const std::string& n)
232
{
233 234
    if(n.empty())
    {
235
#ifdef DEBUG
236
        std::cout << "<<DEBUG>> no data plugin specified, returning a vertial_segment loader" << std::endl;
237
#endif
238 239 240 241
        return new vertical_segment();
    }

#ifdef USING_STATIC
Laurent Belcour's avatar
Laurent Belcour committed
242 243 244 245 246 247 248 249 250 251 252
	 std::string file;
	 if(n[0] == '.')
	 {
		 file = n.substr(1, n.size()-1);
	 }
	 else
	 {
		 file = n;
	 }

    QString path = QDir::currentPath() + QString(file.c_str()) ;
253 254
	 
	 DataPrototype myData = open_library<DataPrototype>(path.toStdString(), "provide_data");
255 256
    if(myData != NULL)
    {
257
#ifdef DEBUG
258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276
        std::cout << "<<DEBUG>> using function provider in file \"" << n << "\"" << std::endl;
#endif
        return myData();
    }
    else
    {
        std::cerr << "<<ERROR>> no data provider found in file \"" << n << "\"" << std::endl;
        return new vertical_segment() ;
    }
#else

    if(_functions.count(n) == 0)
    {
        return new vertical_segment() ;
    }
    else
    {
        return _datas.find(n)->second ;
    }
277
#endif
278
}
279
fitter* plugins_manager::get_fitter(const std::string& n)
280
{
Laurent Belcour's avatar
Laurent Belcour committed
281 282 283 284 285 286 287 288
    if(n.empty())
    {
#ifdef DEBUG
        std::cout << "<<DEBUG>> no fitter plugin specified, returning null" << std::endl;
#endif
        return NULL;
    }

289
#ifdef USING_STATIC
Laurent Belcour's avatar
Laurent Belcour committed
290 291 292 293 294 295 296 297 298 299 300
	 std::string file;
	 if(n[0] == '.')
	 {
		 file = n.substr(1, n.size()-1);
	 }
	 else
	 {
		 file = n;
	 }

    QString path = QDir::currentPath() + QString(file.c_str()) ;
301 302
    
	 FitterPrototype myFitter = open_library<FitterPrototype>(path.toStdString(), "provide_fitter");
Laurent Belcour's avatar
Laurent Belcour committed
303 304 305 306 307 308 309 310 311 312

    if(myFitter != NULL)
    {
#ifdef DEBUG
        std::cout << "<<DEBUG>> using function provider in file \"" << n << "\"" << std::endl;
#endif
        return myFitter();
    }
    else
    {
313 314
        std::cerr << "<<ERROR>> no fitter provider found in file \"" << n << "\"" << std::endl;
        return NULL ;
Laurent Belcour's avatar
Laurent Belcour committed
315 316
    }
#else
317
	if(_fitters.count(n) == 0)
318 319 320 321 322
	{
		return NULL ;
	}
	else
	{
323
#ifdef DEBUG
324
		std::cout << "<<DEBUG>>  using \"" <<  n << "\"" << std::endl ;
325
#endif
Laurent Belcour's avatar
Laurent Belcour committed
326
		return _fitters.find(n)->second ;
327
	}
Laurent Belcour's avatar
Laurent Belcour committed
328
#endif
329
}
330 331 332 333
		
// \todo implement the Darwin (MACOS) version.
#ifdef WIN32
#include <windows.h>
334
size_t plugins_manager::get_system_memory()
335 336 337 338 339 340 341 342
{
	MEMORYSTATUSEX status;
	status.dwLength = sizeof(status);
	GlobalMemoryStatusEx(&status);
	return status.ullTotalPhys;
}
#else
#include <unistd.h>
343
size_t plugins_manager::get_system_memory()
344 345 346 347 348 349
{
	long pages = sysconf(_SC_PHYS_PAGES);
	long page_size = sysconf(_SC_PAGE_SIZE);
	return pages * page_size;
}
#endif