2020-12-22 20:00:51 +01:00
|
|
|
#include <module.h>
|
2020-12-08 04:36:37 +01:00
|
|
|
#include <filesystem>
|
|
|
|
#include <spdlog/spdlog.h>
|
|
|
|
|
|
|
|
ModuleManager::Module_t ModuleManager::loadModule(std::string path) {
|
|
|
|
Module_t mod;
|
2022-02-13 17:25:23 +01:00
|
|
|
|
|
|
|
// On android, the path has to be relative, don't make it absolute
|
|
|
|
#ifndef __ANDROID__
|
2020-12-08 04:36:37 +01:00
|
|
|
if (!std::filesystem::exists(path)) {
|
|
|
|
spdlog::error("{0} does not exist", path);
|
|
|
|
mod.handle = NULL;
|
|
|
|
return mod;
|
|
|
|
}
|
|
|
|
if (!std::filesystem::is_regular_file(path)) {
|
|
|
|
spdlog::error("{0} isn't a loadable module", path);
|
|
|
|
mod.handle = NULL;
|
|
|
|
return mod;
|
|
|
|
}
|
2022-02-13 17:25:23 +01:00
|
|
|
#endif
|
2020-12-08 04:36:37 +01:00
|
|
|
#ifdef _WIN32
|
|
|
|
mod.handle = LoadLibraryA(path.c_str());
|
|
|
|
if (mod.handle == NULL) {
|
2021-02-20 15:27:43 +01:00
|
|
|
spdlog::error("Couldn't load {0}. Error code: {1}", path, GetLastError());
|
2020-12-08 04:36:37 +01:00
|
|
|
mod.handle = NULL;
|
|
|
|
return mod;
|
|
|
|
}
|
|
|
|
mod.info = (ModuleInfo_t*)GetProcAddress(mod.handle, "_INFO_");
|
2021-12-19 22:11:44 +01:00
|
|
|
mod.init = (void (*)())GetProcAddress(mod.handle, "_INIT_");
|
|
|
|
mod.createInstance = (Instance * (*)(std::string)) GetProcAddress(mod.handle, "_CREATE_INSTANCE_");
|
|
|
|
mod.deleteInstance = (void (*)(Instance*))GetProcAddress(mod.handle, "_DELETE_INSTANCE_");
|
|
|
|
mod.end = (void (*)())GetProcAddress(mod.handle, "_END_");
|
2020-12-08 04:36:37 +01:00
|
|
|
#else
|
2021-10-09 18:34:44 +02:00
|
|
|
mod.handle = dlopen(path.c_str(), RTLD_LAZY | RTLD_LOCAL);
|
2020-12-08 04:36:37 +01:00
|
|
|
if (mod.handle == NULL) {
|
|
|
|
spdlog::error("Couldn't load {0}.", path);
|
|
|
|
mod.handle = NULL;
|
|
|
|
return mod;
|
|
|
|
}
|
|
|
|
mod.info = (ModuleInfo_t*)dlsym(mod.handle, "_INFO_");
|
2021-12-19 22:11:44 +01:00
|
|
|
mod.init = (void (*)())dlsym(mod.handle, "_INIT_");
|
|
|
|
mod.createInstance = (Instance * (*)(std::string)) dlsym(mod.handle, "_CREATE_INSTANCE_");
|
|
|
|
mod.deleteInstance = (void (*)(Instance*))dlsym(mod.handle, "_DELETE_INSTANCE_");
|
|
|
|
mod.end = (void (*)())dlsym(mod.handle, "_END_");
|
2020-12-08 04:36:37 +01:00
|
|
|
#endif
|
|
|
|
if (mod.info == NULL) {
|
|
|
|
spdlog::error("{0} is missing _INFO_ symbol", path);
|
|
|
|
mod.handle = NULL;
|
|
|
|
return mod;
|
|
|
|
}
|
|
|
|
if (mod.init == NULL) {
|
|
|
|
spdlog::error("{0} is missing _INIT_ symbol", path);
|
|
|
|
mod.handle = NULL;
|
|
|
|
return mod;
|
|
|
|
}
|
|
|
|
if (mod.createInstance == NULL) {
|
|
|
|
spdlog::error("{0} is missing _CREATE_INSTANCE_ symbol", path);
|
|
|
|
mod.handle = NULL;
|
|
|
|
return mod;
|
|
|
|
}
|
|
|
|
if (mod.deleteInstance == NULL) {
|
|
|
|
spdlog::error("{0} is missing _DELETE_INSTANCE_ symbol", path);
|
|
|
|
mod.handle = NULL;
|
|
|
|
return mod;
|
|
|
|
}
|
|
|
|
if (mod.end == NULL) {
|
|
|
|
spdlog::error("{0} is missing _END_ symbol", path);
|
|
|
|
mod.handle = NULL;
|
|
|
|
return mod;
|
|
|
|
}
|
|
|
|
if (modules.find(mod.info->name) != modules.end()) {
|
|
|
|
spdlog::error("{0} has the same name as an already loaded module", path);
|
|
|
|
mod.handle = NULL;
|
|
|
|
return mod;
|
|
|
|
}
|
|
|
|
for (auto const& [name, _mod] : modules) {
|
|
|
|
if (mod.handle == _mod.handle) {
|
|
|
|
return _mod;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
mod.init();
|
|
|
|
modules[mod.info->name] = mod;
|
|
|
|
return mod;
|
|
|
|
}
|
|
|
|
|
2021-12-16 16:49:19 +01:00
|
|
|
int ModuleManager::createInstance(std::string name, std::string module) {
|
2020-12-08 04:36:37 +01:00
|
|
|
if (modules.find(module) == modules.end()) {
|
|
|
|
spdlog::error("Module '{0}' doesn't exist", module);
|
2021-12-16 16:49:19 +01:00
|
|
|
return -1;
|
2020-12-08 04:36:37 +01:00
|
|
|
}
|
|
|
|
if (instances.find(name) != instances.end()) {
|
|
|
|
spdlog::error("A module instance with the name '{0}' already exists", name);
|
2021-12-16 16:49:19 +01:00
|
|
|
return -1;
|
2020-12-08 04:36:37 +01:00
|
|
|
}
|
|
|
|
int maxCount = modules[module].info->maxInstances;
|
|
|
|
if (countModuleInstances(module) >= maxCount && maxCount > 0) {
|
|
|
|
spdlog::error("Maximum number of instances reached for '{0}'", module);
|
2021-12-16 16:49:19 +01:00
|
|
|
return -1;
|
2020-12-08 04:36:37 +01:00
|
|
|
}
|
|
|
|
Instance_t inst;
|
|
|
|
inst.module = modules[module];
|
|
|
|
inst.instance = inst.module.createInstance(name);
|
|
|
|
instances[name] = inst;
|
2021-07-18 04:30:55 +02:00
|
|
|
onInstanceCreated.emit(name);
|
2021-12-16 16:49:19 +01:00
|
|
|
return 0;
|
2020-12-08 04:36:37 +01:00
|
|
|
}
|
|
|
|
|
2021-12-16 16:49:19 +01:00
|
|
|
int ModuleManager::deleteInstance(std::string name) {
|
2021-05-05 04:31:37 +02:00
|
|
|
if (instances.find(name) == instances.end()) {
|
2021-09-20 19:59:35 +02:00
|
|
|
spdlog::error("Tried to remove non-existent instance '{0}'", name);
|
2021-12-16 16:49:19 +01:00
|
|
|
return -1;
|
2021-05-05 04:31:37 +02:00
|
|
|
}
|
2021-07-18 04:30:55 +02:00
|
|
|
onInstanceDelete.emit(name);
|
2021-12-19 22:11:44 +01:00
|
|
|
Instance_t inst = instances[name];
|
2021-05-05 04:31:37 +02:00
|
|
|
inst.module.deleteInstance(inst.instance);
|
|
|
|
instances.erase(name);
|
2021-07-18 04:30:55 +02:00
|
|
|
onInstanceDeleted.emit(name);
|
2021-12-16 16:49:19 +01:00
|
|
|
return 0;
|
2020-12-08 04:36:37 +01:00
|
|
|
}
|
|
|
|
|
2021-12-16 16:49:19 +01:00
|
|
|
int ModuleManager::deleteInstance(ModuleManager::Instance* instance) {
|
2021-05-05 04:31:37 +02:00
|
|
|
spdlog::error("Delete instance not implemented");
|
2021-12-16 16:49:19 +01:00
|
|
|
return -1;
|
2020-12-08 04:36:37 +01:00
|
|
|
}
|
|
|
|
|
2021-12-16 16:49:19 +01:00
|
|
|
int ModuleManager::enableInstance(std::string name) {
|
2020-12-08 04:36:37 +01:00
|
|
|
if (instances.find(name) == instances.end()) {
|
|
|
|
spdlog::error("Cannot enable '{0}', instance doesn't exist", name);
|
2021-12-16 16:49:19 +01:00
|
|
|
return -1;
|
2020-12-08 04:36:37 +01:00
|
|
|
}
|
|
|
|
instances[name].instance->enable();
|
2021-12-16 16:49:19 +01:00
|
|
|
return 0;
|
2020-12-08 04:36:37 +01:00
|
|
|
}
|
|
|
|
|
2021-12-16 16:49:19 +01:00
|
|
|
int ModuleManager::disableInstance(std::string name) {
|
2020-12-08 04:36:37 +01:00
|
|
|
if (instances.find(name) == instances.end()) {
|
|
|
|
spdlog::error("Cannot disable '{0}', instance doesn't exist", name);
|
2021-12-16 16:49:19 +01:00
|
|
|
return -1;
|
2020-12-08 04:36:37 +01:00
|
|
|
}
|
|
|
|
instances[name].instance->disable();
|
2021-12-16 16:49:19 +01:00
|
|
|
return 0;
|
2020-12-08 04:36:37 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ModuleManager::instanceEnabled(std::string name) {
|
|
|
|
if (instances.find(name) == instances.end()) {
|
|
|
|
spdlog::error("Cannot check if '{0}' is enabled, instance doesn't exist", name);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return instances[name].instance->isEnabled();
|
|
|
|
}
|
|
|
|
|
2021-07-26 03:11:51 +02:00
|
|
|
void ModuleManager::postInit(std::string name) {
|
|
|
|
if (instances.find(name) == instances.end()) {
|
|
|
|
spdlog::error("Cannot post-init '{0}', instance doesn't exist", name);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
instances[name].instance->postInit();
|
|
|
|
}
|
|
|
|
|
2021-07-18 04:30:55 +02:00
|
|
|
std::string ModuleManager::getInstanceModuleName(std::string name) {
|
|
|
|
if (instances.find(name) == instances.end()) {
|
|
|
|
spdlog::error("Cannot get module name of'{0}', instance doesn't exist", name);
|
2021-07-18 04:39:21 +02:00
|
|
|
return "";
|
2021-07-18 04:30:55 +02:00
|
|
|
}
|
|
|
|
return std::string(instances[name].module.info->name);
|
|
|
|
}
|
|
|
|
|
2020-12-08 04:36:37 +01:00
|
|
|
int ModuleManager::countModuleInstances(std::string module) {
|
|
|
|
if (modules.find(module) == modules.end()) {
|
|
|
|
spdlog::error("Cannot count instances of '{0}', Module doesn't exist", module);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
ModuleManager::Module_t mod = modules[module];
|
|
|
|
int count = 0;
|
|
|
|
for (auto const& [name, instance] : instances) {
|
|
|
|
if (instance.module == mod) { count++; }
|
|
|
|
}
|
|
|
|
return count;
|
2021-07-26 03:11:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void ModuleManager::doPostInitAll() {
|
|
|
|
for (auto& [name, inst] : instances) {
|
|
|
|
spdlog::info("Running post-init for {0}", name);
|
|
|
|
inst.instance->postInit();
|
|
|
|
}
|
2020-12-08 04:36:37 +01:00
|
|
|
}
|