8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-27 20:03:03 +01:00
firebird-mirror/src/jrd/plugin_manager.h
bellardo 938ebf794b Added support for a single, generic "plugins" directory. Where possible, this
new plugins directory should be used for engine modules.  Note the old separate
directories intl and UDF still exist and have the same usage as before.
2002-09-24 00:27:23 +00:00

238 lines
9.0 KiB
C++

/*
* plugin_manager.h
* firebird_test
*
* Created by john on Wed Jan 09 2002.
* Copyright (c) 2001 __MyCompanyName__. All rights reserved.
*
*/
#ifndef PLUGIN_MANAGER_H
#define PLUGIN_MANAGER_H
#include "../jrd/os/mod_loader.h"
#include "fb_map.h"
#include "fb_list.h"
#include "fb_string.h"
/** The PluginManager class provides a platform independent interface for
* loading, unloading, and accessing symbols in dynamically loadable
* libraries. The PluginManager has the ability to restrict module
* loading to specific search paths, ignore certain modules in a
* search path, and automatically load all the modules in the search
* paths.
*
* In addition, the PluginManager provides for a single, unified way of
* handling loadable modules. This is to prevent an explosion of
* separate PluginManager objects as the engine grows. The pre-existing
* subsystems that used loadable modules (for example: intl, flu) will still
* have their own PluginManager instance. But other subsystems should try
* to stay within the boundries of the new, more generic, plugin interface.
*
* This interface is divided into two parts: enginge and plugin. The engine
* interface consists of a single function: load_engine_plugins. This
* function searches through the plugin directory (XXX) and loads all the
* plugins it finds. After loading it calls the plugin's "register" function.
* This function has the signature "extern "C" void register_plugin(void *parm)".
* This function is, in turn, responsible for registering the plugin with the
* various parts of the engine as required. To do this each engine subsystem
* that wants to make use of plugins needs to provide a "registration" function.
* This function can take any parameters as necessary to establish the link between
* the plugin and the subsystem, but it must accept the void* parameter given to
* the plugin's register function. This parameter is a Plugin pointer which allows
* the subsystem to access the plugin's symbols.
*
* The exact semantics of the register function the plugin needs to call varies by
* subsystem. Please refer to the subsystem documentation for details.
**/
class PluginManager
{
public:
static void load_engine_plugins();
class iterator;
private:
/** PluginManager::Module is an internal class that encapsulates
* the reference counter, module linked list, module
* name, and symbol lookup function of a loaded module. This class
* is expected to be subclassed to provide module specific functionality.
* For example, a builtin module class may load modules that have been statically
* linked in with the engine, while a dynamic module class might handle modules
* found in OS specific dynamic libraries.
**/
class Module
{
public:
/// The constructor requires the name of the module. The initial reference count
/// is 1, to indicate that the PluginManager itself has a reference. This is
/// verified in debug builds in the PluginManager destructor.
Module(const Firebird::string& name)
: refCnt(1), module_name(name), prev(0), next(0) {}
/// The destructor is responsible for removing the module from the linked
/// list.
virtual ~Module();
/// aquire indicates the interest of a particular piece of engine code in
/// the loadable module. Currently that interest is tracked via a reference
/// counter.
void aquire() { refCnt++; }
/// release indicates the end of interest in a modules. Each call to aquire
/// must be balanced with a call to release.
void release() { refCnt--; }
/// Accessor function that returns the name of the module
const Firebird::string& name() const { return module_name; }
/// lookupSymbol searches through a module's symbol table and attempts to
/// locate the given symbol name. If successful it returns a pointer to
/// the symbol's location in the current address space. If the symbol can't
/// be found it returns 0.
virtual void *lookupSymbol(Firebird::string&) = 0;
private:
friend class PluginManager;
friend class iterator;
int refCnt;
Firebird::string module_name;
Module **prev;
Module *next;
/// unload_module is used called by the destructor when the module
/// represented by this module object needs to be physically unloaded
/// from memory.
virtual void unload_module() {}
};
public:
friend class Plugin;
friend class iterator;
PluginManager() : moduleList(0) {}
~PluginManager();
/** The Plugin class encapsulates the various operations that are available
* after a plugin has been located. The PluginManager returns instances
* of the Plugin class to the calling code to give that code access to
* the aforementioned operations. These operations include tracking the
* usage of a plugin, and locating symbols within the plugin. The Plugin
* class has a full contingent of constructors, assignment operators, and
* equality operators.
**/
class Plugin
{
public:
friend class PluginManager;
Plugin() : module(0) {}
Plugin(const Plugin& src) : module(src.module)
{ if (module) module->aquire(); }
~Plugin() { if (module) module->release(); }
const Plugin& operator=(const Plugin&);
bool operator ==(const Plugin &other) const
{ return module == other.module; }
bool operator !=(const Plugin &other) const
{ return module != other.module; }
/// Looks in the Plugin's symbol table for a symbol with the given name.
/// If the symbol exists a pointer to that symbol's location in the
/// current address space is returned. Otherwise NULL is returned.
void *lookupSymbol(Firebird::string &sym)
{ return module ? module->lookupSymbol(sym) : 0; }
operator bool() { return module != 0; }
private:
friend class iterator;
Plugin(Module *mod) : module(mod)
{ if (module) module->aquire(); }
Module *module;
};
/** The iterator class allows external code to iterate through all the plugins
* a plugin manager knows about. The iterator behaves like a traditional STL
* iterator.
**/
class iterator
{
public:
friend class PluginManager;
const iterator &operator++()
{ if (curr) curr = curr->next; return *this; }
Plugin operator*()
{ return Plugin(curr); }
bool operator==(const iterator& cmp) const
{ return curr == cmp.curr; }
bool operator!=(const iterator& cmp) const
{ return curr != cmp.curr; }
operator bool() const { return curr != 0; }
const iterator &operator=(const iterator& other)
{ curr = other.curr; return *this; }
iterator(const iterator& other) : curr(other.curr) {}
private:
Module *curr;
iterator(Module *start = 0) : curr(start) {}
};
/// Searches for the plugin with the given name using the
/// preset search criteria.
Plugin findPlugin(const Firebird::string&);
/// Adds a path to the list of paths to be searched for a modules. The
/// second parameter indicates if the path is absolute, or relative to
/// the current working directory.
void addSearchPath(const Firebird::string&, bool = true);
/// Removes the given search path from the list of search paths. The path name
/// (1st parameter) and the relative indicator (2nd parameter) must exactly match
/// those passed in to addSearchPath.
void removeSearchPath(const Firebird::string&, bool = true);
/// Adds a module to the ignore list. Ignored modules are not loaded automatically
/// when a directory is scanned, but may be loaded manually via the findPlugin
/// function.
void addIgnoreModule(const Firebird::string &mod)
{ ignoreModules.push_back(mod); }
/// Loads all the plugins found in the current set of search paths, except those that
/// have been ignored. This function must be called after adding all the needed search paths,
/// and before iterating through the plugins.
void loadAllPlugins();
/// The traditional STL iterator end function. Returns an iterator that represents the
/// last element in the iteration.
iterator end() { return iterator(0); }
/// Returns an iterator set to the first element in the iteration set. That set
/// is the set of all plugins known to the PluginManager. This includes all plugins
/// found with the loadAllPluings and findPlugin functions.
iterator begin() { return iterator(moduleList); }
private:
typedef std::pair<Firebird::string, bool> Path;
Module *moduleList;
Firebird::list<Path> searchPaths;
Firebird::list<Firebird::string> ignoreModules;
Module *loadPluginModule(const Firebird::string& name);
Module *loadBuiltinModule(const Firebird::string& name);
class BuiltinModule : public Module
{
public:
BuiltinModule(const Firebird::string& name) : Module(name) {}
private:
Firebird::map<Firebird::string, void*> symbols;
void *lookupSymbol(Firebird::string&);
};
class PluginModule : public Module
{
public:
PluginModule(const Firebird::string &name, ModuleLoader::Module *mod)
: Module(name), module(mod) {}
private:
ModuleLoader::Module *module;
void unload_module();
void *lookupSymbol(Firebird::string&);
};
};
#endif