plugins_manager.cpp 7.52 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 46 47 48 49 50 51 52

#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;
        return NULL;
    }
#endif
}


// Create the object, parse the argument and load all the plugins
53 54 55 56 57 58 59 60 61 62 63 64 65
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());
	}

66
	/*
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
	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
		}
	}
120
	*/
121 122 123 124 125
}

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

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

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

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

    QString path = QDir::currentPath() + QString(file.c_str()) ;
204 205
	 
	 FunctionPrototype myFunction = open_library<FunctionPrototype>(path.toStdString(), "provide_function");
206 207 208 209 210 211 212 213 214 215 216 217 218 219
    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

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

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

    QString path = QDir::currentPath() + QString(file.c_str()) ;
252 253
	 
	 DataPrototype myData = open_library<DataPrototype>(path.toStdString(), "provide_data");
254 255
    if(myData != NULL)
    {
256
#ifdef DEBUG
257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275
        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 ;
    }
276
#endif
277
}
278
fitter* plugins_manager::get_fitter(const std::string& n)
279
{
Laurent Belcour's avatar
Laurent Belcour committed
280 281 282 283 284 285 286 287
    if(n.empty())
    {
#ifdef DEBUG
        std::cout << "<<DEBUG>> no fitter plugin specified, returning null" << std::endl;
#endif
        return NULL;
    }

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

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

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