2021-10-15 22:21:57 +08:00
|
|
|
#ifndef __NASAL_IMPORT_H__
|
|
|
|
#define __NASAL_IMPORT_H__
|
|
|
|
|
2022-07-16 16:53:11 +08:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#define F_OK 0
|
|
|
|
#endif
|
|
|
|
|
2021-10-15 22:21:57 +08:00
|
|
|
class nasal_import
|
|
|
|
{
|
|
|
|
private:
|
2022-04-21 19:45:16 +08:00
|
|
|
bool lib_loaded;
|
2022-01-31 17:22:44 +08:00
|
|
|
nasal_err& nerr;
|
2021-10-16 23:36:43 +08:00
|
|
|
std::vector<std::string> files;
|
2022-07-08 00:42:56 +08:00
|
|
|
std::vector<std::string> envpath;
|
2022-07-06 22:21:01 +08:00
|
|
|
bool imptchk(const nasal_ast&);
|
|
|
|
bool exist(const std::string&);
|
2022-01-31 17:22:44 +08:00
|
|
|
void linker(nasal_ast&,nasal_ast&&);
|
2022-07-06 22:21:01 +08:00
|
|
|
std::string path(const nasal_ast&);
|
|
|
|
nasal_ast fimpt(nasal_ast&);
|
|
|
|
nasal_ast libimpt();
|
2021-10-15 22:21:57 +08:00
|
|
|
nasal_ast load(nasal_ast&,uint16_t);
|
|
|
|
public:
|
2022-07-08 00:42:56 +08:00
|
|
|
nasal_import(nasal_err&);
|
2021-12-23 21:15:50 +08:00
|
|
|
void link(nasal_parse&,const std::string&);
|
2022-07-18 23:54:44 +08:00
|
|
|
const std::vector<std::string>& filelist() const {return files;}
|
2021-10-15 22:21:57 +08:00
|
|
|
};
|
|
|
|
|
2022-07-08 00:42:56 +08:00
|
|
|
nasal_import::nasal_import(nasal_err& e):lib_loaded(false),nerr(e){
|
|
|
|
#ifdef _WIN32
|
|
|
|
char sep=';';
|
|
|
|
#else
|
|
|
|
char sep=':';
|
|
|
|
#endif
|
|
|
|
std::string PATH=getenv("PATH");
|
|
|
|
size_t last=0;
|
|
|
|
size_t pos=PATH.find(sep,last);
|
|
|
|
while(pos!=std::string::npos)
|
|
|
|
{
|
|
|
|
std::string dirpath=PATH.substr(last,pos-last);
|
|
|
|
if(dirpath.length())
|
|
|
|
envpath.push_back(dirpath);
|
|
|
|
last=pos+1;
|
|
|
|
pos=PATH.find(sep,last);
|
|
|
|
}
|
|
|
|
if(last!=PATH.length())
|
2022-07-09 16:24:58 +08:00
|
|
|
envpath.push_back(PATH.substr(last));
|
2022-07-08 00:42:56 +08:00
|
|
|
}
|
|
|
|
|
2022-07-06 22:21:01 +08:00
|
|
|
std::string nasal_import::path(const nasal_ast& node)
|
2021-10-15 22:21:57 +08:00
|
|
|
{
|
2022-07-06 22:21:01 +08:00
|
|
|
if(node[1].type()==ast_callf)
|
|
|
|
return node[1][0].str();
|
|
|
|
std::string fpath=".";
|
|
|
|
for(size_t i=1;i<node.size();++i)
|
|
|
|
#ifndef _WIN32
|
|
|
|
fpath+="/"+node[i].str();
|
|
|
|
#else
|
|
|
|
fpath+="\\"+node[i].str();
|
|
|
|
#endif
|
2022-07-08 00:42:56 +08:00
|
|
|
return fpath+".nas";
|
2022-07-06 22:21:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool nasal_import::imptchk(const nasal_ast& node)
|
|
|
|
{
|
|
|
|
// only these two kinds of node can be recognized as 'import':
|
|
|
|
/*
|
|
|
|
call
|
|
|
|
|_id:import
|
|
|
|
|_callh:stl
|
|
|
|
|_callh:file
|
|
|
|
*/
|
2022-07-08 00:42:56 +08:00
|
|
|
if(node.type()==ast_call && node[0].str()=="import" && node.size()>=2 && node[1].type()==ast_callh)
|
2022-07-06 22:21:01 +08:00
|
|
|
{
|
|
|
|
for(size_t i=1;i<node.size();++i)
|
|
|
|
if(node[i].type()!=ast_callh)
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
2021-10-15 22:21:57 +08:00
|
|
|
/*
|
|
|
|
call
|
2022-01-31 17:22:44 +08:00
|
|
|
|_id:import
|
|
|
|
|_call_func
|
|
|
|
|_string:'filename'
|
2021-10-15 22:21:57 +08:00
|
|
|
*/
|
2022-01-30 23:26:30 +08:00
|
|
|
return (
|
|
|
|
node.type()==ast_call &&
|
|
|
|
node[0].str()=="import" &&
|
2022-07-08 00:42:56 +08:00
|
|
|
node.size()==2 &&
|
2022-01-30 23:26:30 +08:00
|
|
|
node[1].type()==ast_callf &&
|
|
|
|
node[1].size()==1 &&
|
|
|
|
node[1][0].type()==ast_str
|
|
|
|
);
|
2021-10-15 22:21:57 +08:00
|
|
|
}
|
|
|
|
|
2022-07-06 22:21:01 +08:00
|
|
|
bool nasal_import::exist(const std::string& file)
|
2021-10-15 22:21:57 +08:00
|
|
|
{
|
|
|
|
// avoid importing the same file
|
2021-10-16 23:36:43 +08:00
|
|
|
for(auto& fname:files)
|
2021-10-15 22:21:57 +08:00
|
|
|
if(file==fname)
|
|
|
|
return true;
|
2021-10-16 23:36:43 +08:00
|
|
|
files.push_back(file);
|
2021-10-15 22:21:57 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void nasal_import::linker(nasal_ast& root,nasal_ast&& add_root)
|
|
|
|
{
|
|
|
|
// add children of add_root to the back of root
|
2021-10-16 21:08:57 +08:00
|
|
|
for(auto& i:add_root.child())
|
|
|
|
root.add(std::move(i));
|
2021-10-15 22:21:57 +08:00
|
|
|
}
|
|
|
|
|
2022-07-06 22:21:01 +08:00
|
|
|
nasal_ast nasal_import::fimpt(nasal_ast& node)
|
2021-10-15 22:21:57 +08:00
|
|
|
{
|
2021-12-23 21:15:50 +08:00
|
|
|
nasal_lexer lex(nerr);
|
|
|
|
nasal_parse par(nerr);
|
2021-10-15 22:21:57 +08:00
|
|
|
// get filename and set node to ast_null
|
2022-07-06 22:21:01 +08:00
|
|
|
std::string filename=path(node);
|
2021-10-15 22:21:57 +08:00
|
|
|
node.clear();
|
|
|
|
|
|
|
|
// avoid infinite loading loop
|
2022-07-06 22:21:01 +08:00
|
|
|
if(exist(filename))
|
2021-10-16 21:08:57 +08:00
|
|
|
return {0,ast_root};
|
2022-04-21 20:51:47 +08:00
|
|
|
if(access(filename.c_str(),F_OK)==-1)
|
|
|
|
{
|
|
|
|
nerr.err("link","cannot open file <"+filename+">");
|
|
|
|
return {0,ast_root};
|
|
|
|
}
|
2021-10-15 22:21:57 +08:00
|
|
|
|
|
|
|
// start importing...
|
2021-10-16 23:36:43 +08:00
|
|
|
lex.scan(filename);
|
|
|
|
par.compile(lex);
|
|
|
|
nasal_ast tmp=std::move(par.ast());
|
2021-10-15 22:21:57 +08:00
|
|
|
// check if tmp has 'import'
|
2021-10-16 23:36:43 +08:00
|
|
|
return load(tmp,files.size()-1);
|
2021-10-15 22:21:57 +08:00
|
|
|
}
|
|
|
|
|
2022-07-06 22:21:01 +08:00
|
|
|
nasal_ast nasal_import::libimpt()
|
2022-04-21 19:45:16 +08:00
|
|
|
{
|
2022-07-08 00:42:56 +08:00
|
|
|
#ifdef _WIN32
|
|
|
|
#define nalib ".\\lib.nas"
|
|
|
|
#define nastllib ".\\stl\\lib.nas"
|
|
|
|
#define path_nalib "\\lib.nas"
|
|
|
|
#define path_stllib "\\stl\\lib.nas"
|
2022-06-30 23:15:10 +08:00
|
|
|
#else
|
2022-07-08 00:42:56 +08:00
|
|
|
#define nalib "./lib.nas"
|
|
|
|
#define nastllib "./stl/lib.nas"
|
|
|
|
#define path_nalib "/lib.nas"
|
|
|
|
#define path_stllib "/stl/lib.nas"
|
2022-06-30 23:15:10 +08:00
|
|
|
#endif
|
2022-07-08 00:42:56 +08:00
|
|
|
std::vector<std::string> libpath={nalib,nastllib};
|
|
|
|
for(auto& p:envpath)
|
|
|
|
{
|
|
|
|
libpath.push_back(p+path_nalib);
|
|
|
|
libpath.push_back(p+path_stllib);
|
|
|
|
}
|
|
|
|
|
|
|
|
nasal_lexer lex(nerr);
|
|
|
|
nasal_parse par(nerr);
|
2022-04-21 20:51:47 +08:00
|
|
|
std::string filename="";
|
2022-04-21 19:45:16 +08:00
|
|
|
for(auto& i:libpath)
|
2022-04-21 20:51:47 +08:00
|
|
|
if(access(i.c_str(),F_OK)!=-1)
|
2022-04-21 19:45:16 +08:00
|
|
|
{
|
|
|
|
filename=i;
|
|
|
|
break;
|
|
|
|
}
|
2022-04-21 20:51:47 +08:00
|
|
|
if(!filename.length())
|
|
|
|
{
|
|
|
|
std::string paths="";
|
|
|
|
for(auto& i:libpath)
|
|
|
|
paths+=" "+i+"\n";
|
2022-06-30 23:15:10 +08:00
|
|
|
nerr.err("link","cannot find lib file in these paths:\n"+paths);
|
2022-04-21 20:51:47 +08:00
|
|
|
nerr.chkerr();
|
|
|
|
return {0,ast_root};
|
2022-04-21 19:45:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// avoid infinite loading loop
|
2022-07-06 22:21:01 +08:00
|
|
|
if(exist(filename))
|
2022-04-21 19:45:16 +08:00
|
|
|
return {0,ast_root};
|
|
|
|
|
|
|
|
// start importing...
|
|
|
|
lex.scan(filename);
|
|
|
|
par.compile(lex);
|
|
|
|
nasal_ast tmp=std::move(par.ast());
|
|
|
|
// check if tmp has 'import'
|
|
|
|
return load(tmp,files.size()-1);
|
|
|
|
}
|
|
|
|
|
2021-10-15 22:21:57 +08:00
|
|
|
nasal_ast nasal_import::load(nasal_ast& root,uint16_t fileindex)
|
|
|
|
{
|
|
|
|
nasal_ast new_root(0,ast_root);
|
2022-04-21 19:45:16 +08:00
|
|
|
if(!lib_loaded)
|
|
|
|
{
|
2022-07-06 22:21:01 +08:00
|
|
|
linker(new_root,libimpt());
|
2022-04-21 19:45:16 +08:00
|
|
|
lib_loaded=true;
|
|
|
|
}
|
2021-10-16 21:08:57 +08:00
|
|
|
for(auto& i:root.child())
|
2022-07-06 22:21:01 +08:00
|
|
|
if(imptchk(i))
|
|
|
|
linker(new_root,fimpt(i));
|
2021-10-15 22:21:57 +08:00
|
|
|
// add root to the back of new_root
|
|
|
|
nasal_ast file_head(0,ast_file);
|
|
|
|
file_head.set_num(fileindex);
|
2021-10-16 21:08:57 +08:00
|
|
|
new_root.add(std::move(file_head));
|
2021-10-15 22:21:57 +08:00
|
|
|
linker(new_root,std::move(root));
|
|
|
|
return new_root;
|
|
|
|
}
|
|
|
|
|
2021-10-16 21:08:57 +08:00
|
|
|
void nasal_import::link(nasal_parse& parse,const std::string& self)
|
2021-10-15 22:21:57 +08:00
|
|
|
{
|
|
|
|
// initializing
|
2021-10-16 23:36:43 +08:00
|
|
|
files={self};
|
2021-10-15 22:21:57 +08:00
|
|
|
// scan root and import files,then generate a new ast and return to import_ast
|
|
|
|
// the main file's index is 0
|
2021-10-16 21:08:57 +08:00
|
|
|
parse.ast()=load(parse.ast(),0);
|
2021-10-15 22:21:57 +08:00
|
|
|
}
|
|
|
|
|
2020-10-11 20:28:27 +08:00
|
|
|
#endif
|