2025-02-18 04:10:35 +00:00
|
|
|
#include "ModuleLoader.hpp"
|
|
|
|
|
|
|
|
namespace fs = std::filesystem;
|
|
|
|
|
2025-02-23 09:04:50 +00:00
|
|
|
ModuleLoader::ModuleLoader(std::string module_path)
|
2025-02-18 04:10:35 +00:00
|
|
|
{
|
2025-02-23 09:04:50 +00:00
|
|
|
try {
|
2025-02-23 09:10:51 +00:00
|
|
|
_module_path = fs::absolute(module_path).string();
|
|
|
|
if (!_module_path.empty() && _module_path.back() != '/') {
|
|
|
|
_module_path += '/';
|
2025-02-23 09:04:50 +00:00
|
|
|
}
|
|
|
|
} catch (const fs::filesystem_error&) {
|
2025-02-23 09:10:51 +00:00
|
|
|
_module_path = module_path;
|
|
|
|
if (!_module_path.empty() && _module_path.back() != '/') {
|
|
|
|
_module_path += '/';
|
2025-02-23 09:04:50 +00:00
|
|
|
}
|
2025-02-18 04:10:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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 09:10:51 +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::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 {
|
2025-02-23 09:10:51 +00:00
|
|
|
for (const auto& entry : fs::directory_iterator(_module_path)) {
|
2025-02-18 04:10:35 +00:00
|
|
|
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
|
|
|
{
|
2025-02-23 09:10:51 +00:00
|
|
|
std::string module_so_path = _module_path + module_name + ".so";
|
2025-02-18 04:10:35 +00:00
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
std::vector<std::string> missing_symbols;
|
|
|
|
DPMError validate_error = validate_module_interface(module_handle, missing_symbols);
|
|
|
|
if ( validate_error != DPMError::SUCCESS ) {
|
|
|
|
dlclose(module_handle);
|
|
|
|
return validate_error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return validate_error;
|
2025-02-18 04:10:35 +00:00
|
|
|
}
|
|
|
|
|
2025-02-23 09:04:50 +00:00
|
|
|
DPMError ModuleLoader::execute_module(const std::string& module_name, const std::string& command) const
|
2025-02-18 04:10:35 +00:00
|
|
|
{
|
2025-02-24 01:22:35 +00:00
|
|
|
// declare a module_handle
|
|
|
|
void * module_handle;
|
2025-02-23 09:04:50 +00:00
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// attempt to load the module
|
|
|
|
DPMError load_error = load_module( module_name, module_handle );
|
|
|
|
if ( load_error != DPMError::SUCCESS ) {
|
2025-02-23 09:04:50 +00:00
|
|
|
return load_error;
|
2025-02-18 04:10:35 +00:00
|
|
|
}
|
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// Clear any previous error state and handle any residual failure
|
|
|
|
const char* pre_error = dlerror();
|
|
|
|
if ( pre_error != nullptr ) {
|
|
|
|
dlclose( module_handle );
|
|
|
|
return DPMError::UNDEFINED_ERROR;
|
2025-02-23 09:10:51 +00:00
|
|
|
}
|
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// declare a function pointer type to hold the module symbol to execute
|
|
|
|
typedef int (*ExecuteFn) ( const char*, int, char** );
|
2025-02-18 04:10:35 +00:00
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// populate that void pointer to the execute symbol in the module with
|
|
|
|
ExecuteFn execute_fn = (ExecuteFn) dlsym( module_handle, "dpm_module_execute" );
|
2025-02-23 09:04:50 +00:00
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// do basic error handling to detect if the symbol look up was successful
|
|
|
|
const char * dlsym_error = dlerror();
|
|
|
|
if ( dlsym_error != nullptr ) {
|
|
|
|
dlclose( module_handle );
|
|
|
|
return DPMError::SYMBOL_NOT_FOUND;
|
2025-02-23 08:26:49 +00:00
|
|
|
}
|
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// check if the void pointer was populated
|
|
|
|
if ( execute_fn == nullptr ) {
|
|
|
|
dlclose( module_handle );
|
|
|
|
return DPMError::SYMBOL_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
// execute the symbol that was loaded and supply the command string being routed from DPM
|
|
|
|
int exec_error = execute_fn( command.c_str(), 0, nullptr );
|
|
|
|
|
|
|
|
// irregardless of result, this is the time to close the module handle
|
|
|
|
dlclose( module_handle );
|
|
|
|
|
|
|
|
// if the result of execution was not 0, return an error
|
|
|
|
if ( exec_error != 0 ) {
|
|
|
|
return DPMError::SYMBOL_EXECUTION_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if we made it here, assume it was successful
|
|
|
|
return DPMError::SUCCESS;
|
2025-02-23 08:26:49 +00:00
|
|
|
}
|
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
|
|
|
|
DPMError ModuleLoader::get_module_version( void * module_handle, std::string& version ) const
|
2025-02-23 08:26:49 +00:00
|
|
|
{
|
2025-02-24 01:22:35 +00:00
|
|
|
// validate that the module is even loaded
|
|
|
|
if ( !module_handle ) {
|
|
|
|
version = "DPM ERROR";
|
|
|
|
return DPMError::MODULE_NOT_LOADED;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clear any previous error state and handle any residual failure
|
|
|
|
const char* pre_error = dlerror();
|
|
|
|
if ( pre_error != nullptr ) {
|
|
|
|
version = pre_error;
|
|
|
|
return DPMError::UNDEFINED_ERROR;
|
2025-02-23 08:26:49 +00:00
|
|
|
}
|
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// declare a function pointer type to hold the module symbol to execute
|
|
|
|
typedef const char * (* VersionFn)();
|
2025-02-23 08:26:49 +00:00
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// populate that void pointer to execute the symbol in the module with
|
|
|
|
VersionFn version_fn = (VersionFn) dlsym( module_handle, "dpm_module_get_version" );
|
2025-02-23 08:26:49 +00:00
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// Check for errors from dlsym
|
2025-02-23 08:26:49 +00:00
|
|
|
const char* error = dlerror();
|
|
|
|
if (error != nullptr) {
|
2025-02-24 01:22:35 +00:00
|
|
|
version = error;
|
|
|
|
return DPMError::SYMBOL_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if the void pointer was populated
|
|
|
|
if ( version_fn == nullptr ) {
|
|
|
|
version = "ERROR";
|
|
|
|
return DPMError::SYMBOL_NOT_FOUND;
|
2025-02-23 08:26:49 +00:00
|
|
|
}
|
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// execute the loaded symbol
|
|
|
|
const char * ver = version_fn();
|
|
|
|
|
|
|
|
// check the return, and throw an error if it's a null value
|
|
|
|
if ( ver == nullptr ) {
|
|
|
|
version = "MODULE ERROR";
|
|
|
|
return DPMError::INVALID_MODULE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if you made it here, assume success
|
2025-02-23 08:26:49 +00:00
|
|
|
return DPMError::SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
DPMError ModuleLoader::get_module_description(void* module_handle, std::string& description) const
|
|
|
|
{
|
2025-02-24 01:22:35 +00:00
|
|
|
// validate that the module is even loaded
|
2025-02-23 08:26:49 +00:00
|
|
|
if (!module_handle) {
|
2025-02-24 01:22:35 +00:00
|
|
|
description = "DPM ERROR";
|
|
|
|
return DPMError::MODULE_NOT_LOADED;
|
2025-02-23 08:26:49 +00:00
|
|
|
}
|
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// Clear any previous error state and handle any residual failure
|
|
|
|
const char* pre_error = dlerror();
|
|
|
|
if ( pre_error != nullptr ) {
|
|
|
|
version = pre_error;
|
|
|
|
return DPMError::UNDEFINED_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
// declare a function pointer type to hold the module symbol to execute
|
|
|
|
typedef const char * (* DescriptionFn)();
|
2025-02-23 08:26:49 +00:00
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// populate that void pointer to execute the symbol in the module with
|
|
|
|
DescriptionFn description_fn = (DescriptionFn) dlsym( module_handle, "dpm_get_description" );
|
2025-02-23 08:26:49 +00:00
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// check for errors from dlsym
|
2025-02-23 08:26:49 +00:00
|
|
|
const char* error = dlerror();
|
2025-02-24 01:22:35 +00:00
|
|
|
if ( error != nullptr ) {
|
|
|
|
description = "ERROR";
|
|
|
|
return DPMError::SYMBOL_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if the void pointer was populated
|
|
|
|
if ( description_fn == nullptr ) {
|
|
|
|
description = "ERROR";
|
|
|
|
return DPMError::INVALID_MODULE;
|
2025-02-23 08:26:49 +00:00
|
|
|
}
|
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// execute the loaded symbol
|
|
|
|
const char* desc = description_fn();
|
|
|
|
|
|
|
|
// check the return, and throw an error if it's a null value
|
|
|
|
if ( desc == nullptr ) {
|
|
|
|
description = "MODULE ERROR";
|
|
|
|
return DPMError::INVALID_MODULE;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if you made it here, assume success
|
2025-02-23 08:26:49 +00:00
|
|
|
return DPMError::SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
DPMError ModuleLoader::validate_module_interface(void* module_handle, std::vector<std::string>& missing_symbols) const
|
|
|
|
{
|
2025-02-24 01:22:35 +00:00
|
|
|
// validate that the module is even loaded
|
|
|
|
if ( !module_handle ) {
|
|
|
|
return DPMError::MODULE_NOT_LOADED;
|
2025-02-18 04:10:35 +00:00
|
|
|
}
|
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// ensure our starting point of missing symbols is empty
|
2025-02-23 08:26:49 +00:00
|
|
|
missing_symbols.clear();
|
|
|
|
|
2025-02-24 01:22:35 +00:00
|
|
|
// get the size of the loop (should be equal to the number of required symbols)
|
2025-02-23 08:26:49 +00:00
|
|
|
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
|
|
|
}
|