make apps customizable
This commit is contained in:
@@ -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()) {
|
||||||
|
|
||||||
|
|||||||
@@ -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; }
|
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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 \
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -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();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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; }
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -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;
|
||||||
|
|||||||
@@ -11,11 +11,17 @@ namespace Archimedes {
|
|||||||
|
|
||||||
friend class App;
|
friend class App;
|
||||||
|
|
||||||
public:
|
static std::list<std::string*> modules;
|
||||||
typedef Module* create_t(void*, App&);
|
|
||||||
|
public:
|
||||||
|
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;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -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);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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);
|
||||||
|
|||||||
Reference in New Issue
Block a user