This commit is contained in:
Valk Richard Li 2020-11-28 01:14:23 +08:00
parent ffdc8993c4
commit bbf5217374
3 changed files with 143 additions and 230 deletions

View File

@ -16,8 +16,6 @@ void help()
std::cout<<">> [lex ] use lexer to turn code into tokens.\n"; std::cout<<">> [lex ] use lexer to turn code into tokens.\n";
std::cout<<">> [ast ] do parsing and check the abstract syntax tree.\n"; std::cout<<">> [ast ] do parsing and check the abstract syntax tree.\n";
std::cout<<">> [run ] run abstract syntax tree.\n"; std::cout<<">> [run ] run abstract syntax tree.\n";
std::cout<<">> [exec ] generate byte code.\n";
std::cout<<">> [erun ] run byte code.\n";
std::cout<<">> [logo ] print logo of nasal .\n"; std::cout<<">> [logo ] print logo of nasal .\n";
std::cout<<">> [exit ] quit nasal interpreter.\n"; std::cout<<">> [exit ] quit nasal interpreter.\n";
return; return;
@ -107,51 +105,6 @@ void runtime_start()
return; return;
} }
void codegen_start()
{
lexer.openfile(inputfile);
lexer.scanner();
if(lexer.get_error())
{
die("lexer",inputfile);
return;
}
parse.set_toklist(lexer.get_token_list());
parse.main_process();
if(parse.get_error())
{
die("parse",inputfile);
return;
}
import.link(parse.get_root());
if(import.get_error())
{
die("import",inputfile);
return;
}
runtime.set_root(import.get_root());
code_generator.output_exec(inputfile+".naexec",import.get_root());
if(code_generator.get_error())
{
die("code",inputfile);
return;
}
return;
}
void execution_start()
{
code_generator.load_exec(inputfile,import.get_root());
if(code_generator.get_error())
{
die("code",inputfile);
return;
}
runtime.set_root(import.get_root());
runtime.run();
return;
}
int main() int main()
{ {
std::string command; std::string command;
@ -201,10 +154,6 @@ int main()
ast_print(); ast_print();
else if(command=="run") else if(command=="run")
runtime_start(); runtime_start();
else if(command=="exec")
codegen_start();
else if(command=="erun")
execution_start();
else if(command=="logo") else if(command=="logo")
logo(); logo();
else if(command=="exit") else if(command=="exit")

View File

@ -26,7 +26,8 @@ enum op_code
struct opcode struct opcode
{ {
unsigned char op; unsigned char op;
int index; unsigned int index;
opcode(){op=op_nop;index=0;}
}; };
// unfinished // unfinished
@ -35,223 +36,186 @@ struct opcode
class nasal_codegen class nasal_codegen
{ {
private: private:
std::vector<unsigned char> file_header; std::map<double,int> number_table;
std::vector<unsigned char> output_file; std::map<std::string,int> string_table;
std::vector<std::string> string_table; std::vector<opcode> exec_code;
int error; int error;
void init(); void nil_gen();
void output_int(unsigned int,std::vector<unsigned char>&); void number_gen(nasal_ast&);
void output_short(unsigned short,std::vector<unsigned char>&); void string_gen(nasal_ast&);
void output_root(nasal_ast&); void vector_gen(nasal_ast&);
unsigned int input_int(std::ifstream&); void hash_gen(nasal_ast&);
unsigned short input_short(std::ifstream&); void function_gen(nasal_ast&);
void input_root(nasal_ast&,std::ifstream&); void definition_gen(nasal_ast&);
void multi_assignment_gen(nasal_ast&);
void conditional_gen(nasal_ast&);
void loop_gen(nasal_ast&);
void calculation_gen(nasal_ast&);
public: public:
nasal_codegen(); nasal_codegen();
int get_error(); void main_progress(nasal_ast&);
void output_exec(std::string,nasal_ast&);
void load_exec(std::string,nasal_ast&);
}; };
nasal_codegen::nasal_codegen() nasal_codegen::nasal_codegen()
{ {
error=0; error=0;
number_table.clear();
string_table.clear(); string_table.clear();
return; return;
} }
int nasal_codegen::get_error() void nasal_codegen::nil_gen()
{ {
return error; opcode op;
} op.op=op_pushnil;
exec_code.push_back(op);
void nasal_codegen::init()
{
error=0;
string_table.clear();
file_header.clear();
output_file.clear();
return; return;
} }
void nasal_codegen::output_int(unsigned int num,std::vector<unsigned char>& vec) void nasal_codegen::number_gen(nasal_ast& ast)
{ {
unsigned int tmp=0xff000000; int size=number_table.size();
unsigned int offset=24; double num=ast.get_num();
for(int i=0;i<4;++i) opcode op;
if(number_table.find(num)==number_table.end())
number_table[num]=size;
if(num==1)
op.op=op_pushone;
else if(num==0)
op.op=op_pushzero;
else
{ {
vec.push_back((unsigned char)((tmp&num)>>offset)); op.op=op_pushnum;
offset-=8; op.index=number_table[num];
tmp>>=8;
} }
exec_code.push_back(op);
return; return;
} }
void nasal_codegen::output_short(unsigned short num,std::vector<unsigned char>& vec) void nasal_codegen::string_gen(nasal_ast& ast)
{ {
vec.push_back((unsigned char)((num&0xff00)>>8)); int size=string_table.size();
vec.push_back((unsigned char)(num&0x00ff)); std::string str=ast.get_str();
opcode op;
if(string_table.find(str)==string_table.end())
string_table[str]=size;
op.op=op_pushstr;
op.index=string_table[str];
exec_code.push_back(op);
return; return;
} }
void nasal_codegen::output_root(nasal_ast& root) void nasal_codegen::vector_gen(nasal_ast& ast)
{ {
unsigned char type=(unsigned char)root.get_type(); int size=ast.get_children().size();
output_file.push_back(type); opcode op;
std::vector<nasal_ast>& ref_vec=root.get_children(); op.op=op_newvec;
unsigned short root_children_size=ref_vec.size(); op.index=0;
output_short(root_children_size,output_file); exec_code.push_back(op);
if(type==ast_number || type==ast_string || type==ast_identifier || type==ast_dynamic_id || type==ast_call_hash)
{
std::string tmp=root.get_str();
if(type==ast_number)
tmp=trans_number_to_string(root.get_num());
if(std::find(string_table.begin(),string_table.end(),tmp)==string_table.end())
{
string_table.push_back(tmp);
output_short(string_table.size()-1,output_file);
}
else
{
for(int i=0;i<string_table.size();++i)
if(string_table[i]==tmp)
{
output_short(i,output_file);
break;
}
}
}
for(unsigned short i=0;i<root_children_size;++i)
output_root(ref_vec[i]);
return;
}
void nasal_codegen::output_exec(std::string filename,nasal_ast& root)
{
// initializing
this->init();
// put header input file_header
unsigned char header[4]={0x40,0x56,0x4b,0x21};
for(int i=0;i<4;++i)
file_header.push_back(header[i]);
// main progress,put codes into output_file
output_root(root);
// put string table into file_header
output_short(string_table.size(),file_header);
for(int i=0;i<string_table.size();++i)
{
int len=string_table[i].length();
output_int(len,file_header);
for(int j=0;j<len;++j)
file_header.push_back((unsigned char)string_table[i][j]);
}
// output to file
std::ofstream fout(filename,std::ios::binary);
if(fout.fail())
{
++error;
std::cout<<">> [code] cannot generate file \""<<filename<<"\"."<<std::endl;
fout.close();
return;
}
int size=file_header.size();
for(int i=0;i<size;++i) for(int i=0;i<size;++i)
fout<<file_header[i]; {
size=output_file.size(); calculation_gen(ast.get_children()[i]);
op.op=op_vecappend;
op.index=0;
exec_code.push_back(op);
}
return;
}
void nasal_codegen::hash_gen(nasal_ast& ast)
{
int size=ast.get_children().size();
opcode op;
op.op=op_newhash;
op.index=0;
exec_code.push_back(op);
for(int i=0;i<size;++i) for(int i=0;i<size;++i)
fout<<output_file[i];
fout.close();
return;
}
unsigned int nasal_codegen::input_int(std::ifstream& fin)
{
unsigned int number=0;
for(int i=0;i<4;++i)
{ {
number<<=8; string_gen(ast.get_children()[i].get_children()[0]);
number+=(unsigned char)fin.get(); calculation_gen(ast.get_children()[i].get_children()[1]);
} op.op=op_hashappend;
return number; op.index=0;
} exec_code.push_back(op);
unsigned short nasal_codegen::input_short(std::ifstream& fin)
{
unsigned short number=(unsigned char)fin.get();
number=(number<<8)+(unsigned char)fin.get();
return number;
}
void nasal_codegen::input_root(nasal_ast& root,std::ifstream& fin)
{
unsigned char type=(unsigned char)fin.get();
unsigned short size=input_short(fin);
root.set_type(type);
if(type==ast_number || type==ast_string || type==ast_identifier || type==ast_dynamic_id || type==ast_call_hash)
{
std::string tmp=string_table[input_short(fin)];
if(type==ast_number)
root.set_num(trans_string_to_number(tmp));
else
root.set_str(tmp);
}
for(unsigned short i=0;i<size;++i)
{
nasal_ast new_ast;
root.add_child(new_ast);
input_root(root.get_children().back(),fin);
if(error)
return;
} }
return; return;
} }
void nasal_codegen::load_exec(std::string filename,nasal_ast& root) void nasal_codegen::function_gen(nasal_ast& ast)
{ {
// initializing opcode op;
string_table.clear(); op.op=op_newfunc;
root.clear(); op.index=0;
exec_code.push_back(op);
// start input
std::ifstream fin(filename,std::ios::binary); return;
if(fin.fail()) }
void nasal_codegen::definition_gen(nasal_ast& ast)
{
return;
}
void nasal_codegen::multi_assignment_gen(nasal_ast& ast)
{
return;
}
void nasal_codegen::conditional_gen(nasal_ast& ast)
{
return;
}
void nasal_codegen::loop_gen(nasal_ast& ast)
{
return;
}
void nasal_codegen::calculation_gen(nasal_ast& ast)
{
return;
}
void nasal_codegen::main_progress(nasal_ast& ast)
{
int size=ast.get_children().size();
for(int i=0;i<size;++i)
{ {
++error; nasal_ast& tmp=ast.get_children()[i];
std::cout<<">> [code] cannot open file\""<<filename<<"\"."<<std::endl; switch(tmp.get_type())
fin.close();
return;
}
// check header
unsigned char header[4]={0x40,0x56,0x4b,0x21};
for(int i=0;i<4;++i)
if((unsigned char)fin.get()!=header[i])
{ {
++error; case ast_null:
std::cout<<">> [code] \""<<filename<<"\" is not an executable file."<<std::endl; case ast_nil:
return; case ast_number:
case ast_string:
case ast_function:break;
case ast_definition:definition_gen(tmp);break;
case ast_multi_assign:multi_assignment_gen(tmp);break;
case ast_conditional:conditional_gen(tmp);break;
case ast_while:
case ast_for:
case ast_forindex:
case ast_foreach:loop_gen(tmp);break;
case ast_identifier:
case ast_vector:
case ast_hash:
case ast_call:
case ast_equal:
case ast_add_equal:
case ast_sub_equal:
case ast_mult_equal:
case ast_div_equal:
case ast_link_equal:
case ast_unary_sub:
case ast_unary_not:
case ast_add:
case ast_sub:
case ast_mult:
case ast_div:
case ast_link:
case ast_trinocular:calculation_gen(tmp);break;
} }
// input string table
unsigned short string_num=input_short(fin);
for(int i=0;i<string_num;++i)
{
std::string tmp="";
int len=input_int(fin);
for(int j=0;j<len;++j)
{
unsigned char c=(unsigned char)fin.get();
tmp+=c;
}
string_table.push_back(tmp);
} }
// generate root
input_root(root,fin);
return; return;
} }

View File

@ -126,7 +126,7 @@ void nasal_runtime::run()
time_t total_run_time=end_time-begin_time; time_t total_run_time=end_time-begin_time;
if(total_run_time>=1) if(total_run_time>=1)
std::cout<<">> [runtime] process exited after "<<total_run_time<<"s.\n"; std::cout<<">> [runtime] process exited after "<<total_run_time<<"s.\n";
return; return;
} }