94 lines
1.9 KiB
C++
94 lines
1.9 KiB
C++
#pragma once
|
|
|
|
#include <iostream>
|
|
#include <string>
|
|
#include <stdexcept>
|
|
#include <dlfcn.h>
|
|
#include <atomic>
|
|
|
|
using std::cout;
|
|
using std::endl;
|
|
|
|
template<typename T>
|
|
class DynamicLib
|
|
{
|
|
private:
|
|
// IMPORTANT: all dynamic libraries must have produce and destruct methods!
|
|
const std::string produce_name = "produce";
|
|
const std::string destruct_name = "destruct";
|
|
using produce_t = typename T::produce;
|
|
using destruct_t = typename T::destruct;
|
|
|
|
public:
|
|
produce_t produce_method;
|
|
destruct_t destruct_method;
|
|
|
|
DynamicLib(const std::string& lib_path) :
|
|
lib_path(lib_path),
|
|
lib_object(nullptr)
|
|
{
|
|
load();
|
|
}
|
|
|
|
typename T::lib_object* instance()
|
|
{
|
|
// TODO singleton, concurrency
|
|
if (lib_object == nullptr) {
|
|
lib_object = this->produce_method();
|
|
}
|
|
return lib_object;
|
|
}
|
|
|
|
void load()
|
|
{
|
|
load_lib();
|
|
load_produce_func();
|
|
load_destruct_func();
|
|
}
|
|
|
|
~DynamicLib()
|
|
{
|
|
if (lib_object != nullptr) {
|
|
destruct_method(lib_object);
|
|
}
|
|
}
|
|
|
|
private:
|
|
std::string lib_path;
|
|
void *dynamic_lib;
|
|
typename T::lib_object *lib_object;
|
|
|
|
void load_lib()
|
|
{
|
|
dynamic_lib = dlopen(lib_path.c_str(), RTLD_LAZY);
|
|
if (!dynamic_lib) {
|
|
throw std::runtime_error(dlerror());
|
|
}
|
|
dlerror();
|
|
}
|
|
|
|
void load_produce_func()
|
|
{
|
|
produce_method = (produce_t) dlsym(
|
|
dynamic_lib,
|
|
produce_name.c_str()
|
|
);
|
|
const char* dlsym_error = dlerror();
|
|
if (dlsym_error) {
|
|
throw std::runtime_error(dlsym_error);
|
|
}
|
|
}
|
|
|
|
void load_destruct_func()
|
|
{
|
|
destruct_method = (destruct_t) dlsym(
|
|
dynamic_lib,
|
|
destruct_name.c_str()
|
|
);
|
|
const char *dlsym_error = dlerror();
|
|
if (dlsym_error) {
|
|
throw std::runtime_error(dlsym_error);
|
|
}
|
|
}
|
|
};
|