make apps customizable

This commit is contained in:
2025-03-29 15:02:27 -05:00
parent aff69960a0
commit 0d61a369fc
11 changed files with 49 additions and 241 deletions

View File

@@ -1,148 +1,24 @@
#include "App.h" #include "MinimalApp.h"
Archimedes::App* Archimedes::App::instance = nullptr;
namespace Archimedes { namespace Archimedes {
App::App(const int& argc, char* argv[]) { MinimalApp::MinimalApp() : App() {
if(instance != nullptr) {
std::cout << "App already exists\nThere can only be one!\n";
std::abort();
}
std::cout << "Initializing...\n";
instance = this;
handleArgs(argc, argv);
} }
App::~App() { MinimalApp::~MinimalApp() {
std::cout << "\nExiting...\n";
for(auto it = modules.begin(); it != modules.end(); it++) {
void* handle = (*it)->getHandle();
delete *it;
dlclose(handle);
it = modules.erase(it);
}
}
bool App::load(std::string lib, std::list<std::string> blacklist = {}) {
std::cout << "print twice!\n";
void* h = dlopen(lib.c_str(), RTLD_NOW);
if(!h) {
std::cout << "could not open lib at: \"" << lib.c_str() << "\"\nError: " << dlerror() << std::endl;
return false;
}
Module::create_t* create = (Module::create_t*) dlsym(h, "create");
char* err = dlerror();
if(err) {
std::cout << "error finding create function in file: " << lib << std::endl;
std::cout << "dlerror(): " << err << std::endl;
}
Module* m = create(h, App::Get());
for(auto it = blacklist.begin(); it != blacklist.end(); it++) {
if(*it == m->getName()) {
std::cout << "Module \"" << *it << "\" is already loaded!\n";
delete m;
dlclose(h);
return false;
}
}
blacklist.push_back(m->getName());
bool skip = false;
for(auto it = m->deps.begin(); it != m->deps.end(); it++) {
for(std::string s : blacklist) {
if(it->first == s)
skip = true;
}
if(skip) {
skip = false;
continue;
} else {
load(it->second, blacklist);
}
}
modules.push_back(m);
m->setSelf(--modules.end());
modules.end()++;
return true;
}
void App::unload(std::list<Module*>::iterator it) {
Module* m = *it;
void* h = m->getHandle();
modules.erase(m->self);
delete m;
dlclose(h);
} }
void App::stopModule(std::list<Module*>::iterator it) { void MinimalApp::run() {
toClose.push_back(*it);
}
void App::startModule(std::string lib) {
toOpen.push_back(lib);
}
void App::handleArgs(const int& argc, char* argv[]) {
int i = 1;
if(argc == 1) {
printHelp();
end();
}
while(i < argc) {
if(strcmp(argv[i], "-h") == 0 || strcmp(argv[i], "--help") == 0) {
printHelp();
end();
} else {
break;
}
i++;
}
while(i < argc) {
std::cout << "Attempting to load: " << argv[i] << std::endl;
load(argv[i]);
i++;
}
}
void App::printHelp() {
std::cout << "archimedes [ -h | --help ] /path/to/some/library.so\n";
}
void App::run() {
std::cout << "\nTesting...\n"; std::cout << "\nTesting...\n";
for(auto* m : modules) for(auto* m : modules)
m->onLoad(); m->onLoad();
for(auto* s : Archimedes::Module::GetModules())
std::cout << "Module: " << (s ? *s : "nullptr") << std::endl;
// Main loop // Main loop
while (!done && !modules.empty()) { while (!done && !modules.empty()) {

View File

@@ -1,41 +1,23 @@
#ifndef APP_H #ifndef MINIMALAPP_H
#define APP_H #define MINIMALAPP_H
#include "pch.hpp" #include "pch.hpp"
#include "utils/App/App.h"
#include "utils/Module/Module.h" #include "utils/Module/Module.h"
namespace Archimedes { namespace Archimedes {
class App { class MinimalApp : public App {
private: private:
static App* instance;
bool done = false; void handleArgs(const int&, char*[]) {};
std::list<Module*> modules; void printHelp() {};
std::list<Module*> toClose;
std::list<std::string> toOpen;
bool load(std::string, std::list<std::string>);
void unload(std::list<Module*>::iterator);
void handleArgs(const int&, char*[]);
void printHelp();
std::list<std::string> getBlacklist() {
std::list<std::string> l;
for(Module* m : modules)
l.push_back(m->getName());
return l;
}
public: public:
App(const int&, char*[]); MinimalApp();
~App(); ~MinimalApp();
static App& Get() { return *instance; }
void run(); void run();
@@ -43,7 +25,6 @@ namespace Archimedes {
void startModule(std::string); void startModule(std::string);
void end() { done = true; }
}; };
} }

View File

@@ -84,7 +84,8 @@
buildPhase = '' buildPhase = ''
clang++ \ clang++ \
modules/Print/src/*.cpp src/App.cpp \ modules/Print/src/*.cpp \
utils/App/App.cpp \
-fpic -shared \ -fpic -shared \
-I src -I include \ -I src -I include \
-Wall \ -Wall \

View File

@@ -3,6 +3,6 @@
#include "utils/Module/Module.h" #include "utils/Module/Module.h"
#include "utils/GuiModule/GuiModule.h" #include "utils/GuiModule/GuiModule.h"
#include "App.h" #include "utils/App/App.h"
#endif #endif

View File

@@ -5,7 +5,7 @@ Archimedes::App* Archimedes::App::instance = nullptr;
namespace Archimedes { namespace Archimedes {
App::App(const int& argc, char* argv[]) { App::App() {
if(instance != nullptr) { if(instance != nullptr) {
std::cout << "App already exists\nThere can only be one!\n"; std::cout << "App already exists\nThere can only be one!\n";
@@ -16,8 +16,6 @@ namespace Archimedes {
instance = this; instance = this;
handleArgs(argc, argv);
} }
App::~App() { App::~App() {
@@ -34,7 +32,6 @@ namespace Archimedes {
bool App::load(std::string lib, std::list<std::string> blacklist = {}) { bool App::load(std::string lib, std::list<std::string> blacklist = {}) {
std::cout << "print twice!\n"; std::cout << "print twice!\n";
void* h = dlopen(lib.c_str(), RTLD_NOW); void* h = dlopen(lib.c_str(), RTLD_NOW);
@@ -105,61 +102,4 @@ namespace Archimedes {
toOpen.push_back(lib); toOpen.push_back(lib);
} }
void App::handleArgs(const int& argc, char* argv[]) {
int i = 1;
if(argc == 1) {
printHelp();
end();
}
while(i < argc) {
if(strcmp(argv[i], "-h") == 0 || strcmp(argv[i], "--help") == 0) {
printHelp();
end();
} else {
break;
}
i++;
}
while(i < argc) {
std::cout << "Attempting to load: " << argv[i] << std::endl;
load(argv[i]);
i++;
}
}
void App::printHelp() {
std::cout << "archimedes [ -h | --help ] /path/to/some/library.so\n";
}
void App::run() {
std::cout << "\nTesting...\n";
for(auto* m : modules)
m->onLoad();
// Main loop
while (!done && !modules.empty()) {
for(auto* m : modules) {
m->run();
}
for(auto it = toClose.begin(); it != toClose.end(); it++) {
unload(it);
}
toClose.clear();
for(std::string s : toOpen) {
load(s, getBlacklist());
}
toOpen.clear();
}
}
} }

View File

@@ -20,9 +20,9 @@ namespace Archimedes {
virtual bool load(std::string, std::list<std::string>); virtual bool load(std::string, std::list<std::string>);
virtual void unload(std::list<Module*>::iterator); virtual void unload(std::list<Module*>::iterator);
virtual void handleArgs(const int&, char*[]); virtual void handleArgs(const int&, char*[]) = 0;
virtual void printHelp(); virtual void printHelp() = 0;
std::list<std::string> getBlacklist() { std::list<std::string> getBlacklist() {
std::list<std::string> l; std::list<std::string> l;
@@ -32,16 +32,16 @@ namespace Archimedes {
} }
public: public:
App(const int&, char*[]); App();
~App(); virtual ~App();
static App& Get() { return *instance; } static App* Get() { return instance; }
void run(); virtual void run() = 0;
void stopModule(std::list<Module*>::iterator); virtual void stopModule(std::list<Module*>::iterator);
void startModule(std::string); virtual void startModule(std::string);
void end() { done = true; } void end() { done = true; }
}; };

View File

@@ -9,9 +9,9 @@ namespace Archimedes {
class GuiModule : public Module { class GuiModule : public Module {
public: public:
typedef GuiModule* create_t(void*, App&); typedef GuiModule* create_t(void*, App*);
GuiModule(void* h, App& a) : Module(h, a) {} GuiModule(void* h, App* a) : Module(h, a) {}
virtual ~GuiModule() { if(window) delete window; } virtual ~GuiModule() { if(window) delete window; }
virtual void onLoad() = 0; virtual void onLoad() = 0;
virtual void run() = 0; virtual void run() = 0;

View File

@@ -11,11 +11,17 @@ namespace Archimedes {
friend class App; friend class App;
static std::list<std::string*> modules;
public: public:
typedef Module* create_t(void*, App&); typedef Module* create_t(void*, App*);
static std::list<std::string*>& GetModules() { return modules; }
Module(void* h, App* a) : handle(h), app(a) { modules.push_back(&name); }
virtual ~Module() { modules.remove_if([this](std::string* s) -> bool { return s ? *s == name : false; }); }
Module(void* h, App& a) : handle(h), app(a) {};
virtual ~Module() {}
virtual void run() = 0; virtual void run() = 0;
virtual void onLoad() = 0; virtual void onLoad() = 0;
@@ -29,7 +35,7 @@ namespace Archimedes {
void* handle; void* handle;
std::list<Module*>::iterator self; std::list<Module*>::iterator self;
App& app; App* app;
std::unordered_map<std::string, std::string> deps; std::unordered_map<std::string, std::string> deps;
}; };

View File

@@ -1,6 +1,6 @@
#include "Print.h" #include "Print.h"
Print::Print(void* h, App& a) : Module(h, a) { Print::Print(void* h, Archimedes::App* a) : Archimedes::Module(h, a) {
name = "Print"; name = "Print";
} }
@@ -10,5 +10,9 @@ Print::~Print() {
void Print::run() { void Print::run() {
std::cout << "Print lib loaded and run!\n"; std::cout << "Print lib loaded and run!\n";
app.stopModule(self);
for(auto m : Archimedes::Module::GetModules())
std::cout << m << std::endl;
app->stopModule(self);
} }

View File

@@ -1,9 +1,9 @@
#include "../../../include/Archimedes.h" #include "Archimedes.h"
class Print : public Module { class Print : public Archimedes::Module {
public: public:
Print(void*, App&); Print(void*, Archimedes::App*);
~Print(); ~Print();
void run(); void run();
void onLoad() {} void onLoad() {}
@@ -11,7 +11,7 @@ class Print : public Module {
}; };
extern "C" { extern "C" {
Module* create(void* handle, App& app) { Archimedes::Module* create(void* handle, Archimedes::App* app) {
return new Print(handle, app); return new Print(handle, app);
} }
} }

View File

@@ -1,5 +1,5 @@
#include "pch.hpp" #include "pch.hpp"
#include "App.h" #include "utils/App/App.h"
int main(int argc, char* argv[]) { int main(int argc, char* argv[]) {
Archimedes::App app(argc, argv); Archimedes::App app(argc, argv);