You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

162 lines
3.6 KiB

#include "globals.h"
// Variables definitions
G_varsType G_vars;
// Functions addresses
G_funcsType G_funcs;
// List of objects to save
G_toType G_tosave;
// List of objects to print
G_toType G_toprint;
// Loaded modules
G_libsType G_libs;
void ClearGlobals()
{
for(auto& it:G_vars) delete it.second;
for(auto& it:G_tosave) delete it;
for(auto& it:G_toprint) delete it;
for(auto& it:G_libs) dlclose(it);
G_vars.clear();
G_tosave.clear();
G_toprint.clear();
}
int LoadModule(const std::string& name, const void* p, const std::string& modname)
{
const std::string spath=*(reinterpret_cast<const std::string*>(p));
// Load module
void* handle;
void* initfunc;
std::string initname;
if(0!=modname.size()) initname=modname;
else
{
// Remove directory name, if present
initname=name.substr((name.rfind('/')!=std::string::npos)?(name.rfind('/')+1):0,std::string::npos);
// Remove ".so" on the end of string
if(initname.rfind(".so")!=std::string::npos) initname.erase(initname.rfind(".so"),std::string::npos);
}
initname+="_module_init";
// Check if module is statically linked or already loaded: dlopen'ed the main program
handle=dlopen(0,RTLD_LAZY|RTLD_GLOBAL);
if(0==handle)
{
COUT(ERROR)<<std::endl<<"Something wrong: can't dlopen the main program"<<std::endl;
return 1;
}
initfunc=dlsym(handle,initname.c_str());
if(0==initfunc) // We not find it
{
if('/'==name[0]) // Absolute path
{
handle=dlopen(name.c_str(),RTLD_LAZY|RTLD_GLOBAL);
if(0==handle) handle=dlopen((name+".so").c_str(),RTLD_LAZY|RTLD_GLOBAL);
}
else // Relative path
{
std::string curpath;
size_t bpos=0,epos;
// Check in spath
do
{
epos=spath.find(':',bpos);
curpath=spath.substr(bpos,(std::string::npos==epos)?epos:(epos-bpos));
handle=dlopen((curpath+name).c_str(),RTLD_LAZY|RTLD_GLOBAL);
if(0!=handle) break;
handle=dlopen((curpath+name+".so").c_str(),RTLD_LAZY|RTLD_GLOBAL);
if(0!=handle) break;
bpos=epos+1;
} while(std::string::npos!=epos);
}
if(0==handle)
{
COUT(ERROR)<<std::endl<<"Can't dlopen module "<<name<<std::endl;
return 1;
}
G_libs.push_back(handle);
initfunc=dlsym(handle,initname.c_str());
}
if(0==initfunc)
{
COUT(ERROR)<<std::endl<<"Can't find initialising function "<<initname<<" in module "<<name<<std::endl;
return 1;
}
int ret=(*reinterpret_cast<ModuleInitFunc>(initfunc))(p);
if(0!=ret) COUT(ERROR)<<std::endl<<"Initialising function "<<initname<<" of module "<<name<<" returns error "<<ret<<std::endl;
return ret;
}
void RegisterFunction(const std::string& name, Func func)
{
G_funcs.emplace(name,func);
}
bool Save(const ObjectList* input)
{
ObjectList::ListValues::size_type sz=input->Size(), i;
if(sz<2 || sz%2==1 )
{
COUT(ERROR)<<"Number of save arguments must not be "<<sz<<std::endl;
return false;
}
// Check arguments types
for(i=0;i<sz/2;i++)
{
if(!OBType<ObjectString>(input->At(i*2+1)))
{
COUT(ERROR)<<"Save format is save(object_1,file_1,...,object_n,file_n) where file_i is string"<<std::endl;
return false;
}
}
// Save
const ObjectBase* arg1;
for(i=0;i<sz/2;i++)
{
arg1=input->At(i*2);
OBType<ObjectString> arg2(input->At(i*2+1));
if(!arg1->Save(arg2->Value().c_str()))
{
COUT(ERROR)<<"Can't save object "<<arg1->Dump()<<" to file "<<arg2->Value()<<std::endl;
return false;
}
}
return true;
}
bool Print(const ObjectList* input)
{
ObjectList::ListValues::size_type sz=input->Size(), i;
if(sz==0) return true;
// Print
for(i=0;i<sz;i++)
{
if(!input->At(i)->Print())
{
COUT(ERROR)<<"Unprintable object "<<input->At(i)->Dump()<<std::endl;
return false;
}
}
return true;
}