2013-03-05 07:32:27 +01:00
|
|
|
#ifndef REGISTRY_H
|
|
|
|
#define REGISTRY_H
|
|
|
|
|
|
|
|
#include <list>
|
|
|
|
#include <map>
|
2013-03-05 07:36:32 +01:00
|
|
|
#include <string>
|
2013-03-05 07:32:27 +01:00
|
|
|
|
|
|
|
namespace encfs {
|
|
|
|
|
|
|
|
template <typename T>
|
2013-10-20 00:35:26 +02:00
|
|
|
class Registry {
|
|
|
|
public:
|
2013-03-05 07:32:27 +01:00
|
|
|
typedef T *(*FactoryFn)();
|
|
|
|
struct Data {
|
|
|
|
FactoryFn constructor;
|
|
|
|
typename T::Properties properties;
|
|
|
|
};
|
|
|
|
|
2013-10-20 00:35:26 +02:00
|
|
|
void Register(const char *name, FactoryFn fn,
|
|
|
|
typename T::Properties properties) {
|
2013-03-05 07:32:27 +01:00
|
|
|
Data d;
|
|
|
|
d.constructor = fn;
|
|
|
|
d.properties = properties;
|
|
|
|
data[name] = d;
|
|
|
|
}
|
|
|
|
|
2013-10-20 00:35:26 +02:00
|
|
|
T *Create(const char *name) {
|
2013-03-05 07:32:27 +01:00
|
|
|
auto it = data.find(name);
|
2013-10-20 00:35:26 +02:00
|
|
|
if (it == data.end()) return NULL;
|
2013-03-05 07:32:27 +01:00
|
|
|
return (*it->second.constructor)();
|
|
|
|
}
|
|
|
|
|
2013-10-20 00:35:26 +02:00
|
|
|
T *CreateForMatch(const std::string &description) {
|
2013-03-05 07:32:27 +01:00
|
|
|
for (auto &it : data) {
|
2013-03-05 07:36:32 +01:00
|
|
|
auto name = it.second.properties.toString();
|
|
|
|
if (!name.compare(0, description.size(), description))
|
2013-03-05 07:32:27 +01:00
|
|
|
return (*it.second.constructor)();
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::list<std::string> GetAll() const {
|
|
|
|
std::list<std::string> result;
|
|
|
|
for (auto &it : data) {
|
|
|
|
result.push_back(it.first);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
const typename T::Properties *GetProperties(const char *name) const {
|
|
|
|
auto it = data.find(name);
|
2013-10-20 00:35:26 +02:00
|
|
|
if (it == data.end()) return NULL;
|
2013-03-05 07:32:27 +01:00
|
|
|
return &(it->second.properties);
|
|
|
|
}
|
2013-10-20 00:35:26 +02:00
|
|
|
|
2013-03-05 07:36:32 +01:00
|
|
|
const typename T::Properties *GetPropertiesForMatch(
|
|
|
|
const std::string &description) const {
|
|
|
|
for (auto &it : data) {
|
|
|
|
auto name = it.second.properties.toString();
|
|
|
|
if (!name.compare(0, description.size(), description))
|
|
|
|
return &(it.second.properties);
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-10-20 00:35:26 +02:00
|
|
|
private:
|
2013-03-05 07:32:27 +01:00
|
|
|
std::map<std::string, Data> data;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename T, typename BASE>
|
2013-10-20 00:35:26 +02:00
|
|
|
class Registrar {
|
|
|
|
public:
|
|
|
|
Registrar(const char *name) {
|
|
|
|
BASE::GetRegistry().Register(name, Registrar<T, BASE>::Construct,
|
2013-03-05 07:32:27 +01:00
|
|
|
T::GetProperties());
|
|
|
|
}
|
|
|
|
|
2013-10-20 00:35:26 +02:00
|
|
|
static BASE *Construct() { return new T(); }
|
2013-03-05 07:32:27 +01:00
|
|
|
};
|
|
|
|
|
2013-10-20 00:35:26 +02:00
|
|
|
#define DECLARE_REGISTERABLE_TYPE(TYPE) static Registry<TYPE> &GetRegistry()
|
2013-03-05 07:32:27 +01:00
|
|
|
|
2013-10-20 00:35:26 +02:00
|
|
|
#define DEFINE_REGISTERABLE_TYPE(TYPE) \
|
|
|
|
Registry<TYPE> &TYPE::GetRegistry() { \
|
|
|
|
static Registry<TYPE> registry; \
|
|
|
|
return registry; \
|
|
|
|
}
|
2013-03-05 07:32:27 +01:00
|
|
|
|
|
|
|
#define REGISTER_CLASS(DERIVED, BASE) \
|
2013-10-20 00:35:26 +02:00
|
|
|
static Registrar<DERIVED, BASE> registrar_##DERIVED##_##BASE(#DERIVED)
|
2013-03-05 07:32:27 +01:00
|
|
|
|
|
|
|
} // namespace encfs
|
|
|
|
|
2013-10-20 00:35:26 +02:00
|
|
|
#endif // REGISTRY_H
|