2025-02-18 04:10:35 +00:00
|
|
|
#include "ModuleLoader.hpp"
|
|
|
|
|
|
|
|
namespace fs = std::filesystem;
|
|
|
|
|
|
|
|
ModuleLoader::ModuleLoader(std::string module_path) : module_path_(std::move(module_path))
|
|
|
|
{
|
|
|
|
if (!module_path_.empty() && module_path_.back() != '/') {
|
|
|
|
module_path_ += '/';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-02-23 08:26:49 +00:00
|
|
|
DPMError ModuleLoader::get_module_path(std::string& path) const
|
2025-02-18 04:10:35 +00:00
|
|
|
{
|
2025-02-23 08:26:49 +00:00
|
|
|
path = module_path_;
|
2025-02-18 04:10:35 +00:00
|
|
|
return DPMError::SUCCESS;
|
|
|
|
}
|
|
|
|
|
2025-02-23 08:26:49 +00:00
|
|
|
DPMError ModuleLoader::get_absolute_module_path(std::string& abs_path) const
|
2025-02-18 04:10:35 +00:00
|
|
|
{
|
|
|
|
try {
|
2025-02-23 08:26:49 +00:00
|
|
|
abs_path = fs::absolute(module_path_).string();
|
|
|
|
return DPMError::SUCCESS;
|
2025-02-18 04:10:35 +00:00
|
|
|
} catch (const fs::filesystem_error&) {
|
2025-02-23 08:26:49 +00:00
|
|
|
abs_path = module_path_;
|
|
|
|
return DPMError::PATH_NOT_FOUND;
|
2025-02-18 04:10:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-02-23 08:26:49 +00:00
|
|
|
DPMError ModuleLoader::list_available_modules(std::vector<std::string>& modules) const
|
2025-02-18 04:10:35 +00:00
|
|
|
{
|
2025-02-23 08:26:49 +00:00
|
|
|
modules.clear();
|
2025-02-18 04:10:35 +00:00
|
|
|
|
|
|
|
try {
|
|
|
|
fs::path absolute_path = fs::absolute(module_path_);
|
|
|
|
for (const auto& entry : fs::directory_iterator(absolute_path)) {
|
|
|
|
if (entry.is_regular_file()) {
|
|
|
|
std::string filename = entry.path().filename().string();
|
|
|
|
if (filename.size() > 3 && filename.substr(filename.size() - 3) == ".so") {
|
|
|
|
modules.push_back(filename.substr(0, filename.size() - 3));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (const fs::filesystem_error&) {
|
2025-02-23 08:26:49 +00:00
|
|
|
return DPMError::PERMISSION_DENIED;
|
2025-02-18 04:10:35 +00:00
|
|
|
}
|
|
|
|
|
2025-02-23 08:26:49 +00:00
|
|
|
return DPMError::SUCCESS;
|
2025-02-18 04:10:35 +00:00
|
|
|
}
|
|
|
|
|
2025-02-23 08:26:49 +00:00
|
|
|
DPMError ModuleLoader::load_module(const std::string& module_name, void*& module_handle) const
|
2025-02-18 04:10:35 +00:00
|
|
|
{
|
|
|
|
std::string module_so_path = module_path_ + module_name + ".so";
|
|
|
|
|
2025-02-23 08:26:49 +00:00
|
|
|
module_handle = dlopen(module_so_path.c_str(), RTLD_LAZY);
|
2025-02-18 04:10:35 +00:00
|
|
|
if (!module_handle) {
|
2025-02-23 08:26:49 +00:00
|
|
|
return DPMError::MODULE_LOAD_FAILED;
|
2025-02-18 04:10:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
dlerror();
|
2025-02-23 08:26:49 +00:00
|
|
|
return DPMError::SUCCESS;
|
2025-02-18 04:10:35 +00:00
|
|
|
}
|
|
|
|
|
2025-02-23 08:26:49 +00:00
|
|
|
DPMError ModuleLoader::execute_module(void* module_handle, const std::string& command) const
|
2025-02-18 04:10:35 +00:00
|
|
|
{
|
|
|
|
if (!module_handle) {
|
2025-02-23 08:26:49 +00:00
|
|
|
return DPMError::INVALID_MODULE;
|
2025-02-18 04:10:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
using ExecuteFn = int (*)(const char*, int, char**);
|
|
|
|
ExecuteFn execute_fn = (ExecuteFn)dlsym(module_handle, "dpm_module_execute");
|
|
|
|
|
|
|
|
const char* error = dlerror();
|
|
|
|
if (error != nullptr) {
|
2025-02-23 08:26:49 +00:00
|
|
|
return DPMError::MODULE_LOAD_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
execute_fn(command.c_str(), 0, nullptr);
|
|
|
|
return DPMError::SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
DPMError ModuleLoader::get_module_version(void* module_handle, std::string& version) const
|
|
|
|
{
|
|
|
|
if (!module_handle) {
|
|
|
|
version = "ERROR";
|
|
|
|
return DPMError::INVALID_MODULE;
|
|
|
|
}
|
|
|
|
|
|
|
|
dlerror();
|
|
|
|
|
|
|
|
using GetVersionFn = const char* (*)();
|
|
|
|
GetVersionFn get_version = (GetVersionFn)dlsym(module_handle, "dpm_module_get_version");
|
|
|
|
|
|
|
|
const char* error = dlerror();
|
|
|
|
if (error != nullptr) {
|
|
|
|
version = "unknown";
|
|
|
|
return DPMError::MODULE_LOAD_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* ver = get_version();
|
|
|
|
version = ver ? ver : "unknown";
|
|
|
|
return DPMError::SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
DPMError ModuleLoader::get_module_description(void* module_handle, std::string& description) const
|
|
|
|
{
|
|
|
|
if (!module_handle) {
|
|
|
|
description = "ERROR";
|
|
|
|
return DPMError::INVALID_MODULE;
|
|
|
|
}
|
|
|
|
|
|
|
|
dlerror();
|
|
|
|
|
|
|
|
using GetDescriptionFn = const char* (*)();
|
|
|
|
GetDescriptionFn get_description = (GetDescriptionFn)dlsym(module_handle, "dpm_get_description");
|
|
|
|
|
|
|
|
const char* error = dlerror();
|
|
|
|
if (error != nullptr) {
|
|
|
|
description = "unknown";
|
|
|
|
return DPMError::MODULE_LOAD_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* desc = get_description();
|
|
|
|
description = desc ? desc : "unknown";
|
|
|
|
return DPMError::SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
DPMError ModuleLoader::validate_module_interface(void* module_handle, std::vector<std::string>& missing_symbols) const
|
|
|
|
{
|
|
|
|
if (!module_handle) {
|
|
|
|
return DPMError::INVALID_MODULE;
|
2025-02-18 04:10:35 +00:00
|
|
|
}
|
|
|
|
|
2025-02-23 08:26:49 +00:00
|
|
|
missing_symbols.clear();
|
|
|
|
|
|
|
|
size_t num_symbols = module_interface::required_symbols.size();
|
|
|
|
for (size_t i = 0; i < num_symbols; i++) {
|
|
|
|
dlerror();
|
|
|
|
void* sym = dlsym(module_handle, module_interface::required_symbols[i].c_str());
|
|
|
|
const char* error = dlerror();
|
|
|
|
|
|
|
|
if (error != nullptr) {
|
|
|
|
missing_symbols.push_back(module_interface::required_symbols[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (missing_symbols.empty()) {
|
|
|
|
return DPMError::SUCCESS;
|
|
|
|
}
|
2025-02-18 04:10:35 +00:00
|
|
|
|
2025-02-23 08:26:49 +00:00
|
|
|
return DPMError::INVALID_MODULE;
|
2025-02-18 04:10:35 +00:00
|
|
|
}
|