add upvalue info into detail crash info
This commit is contained in:
parent
bbee31ea55
commit
885b57cd52
53
nasal_gc.h
53
nasal_gc.h
|
@ -106,13 +106,13 @@ struct nasal_hash// 56 bytes
|
||||||
|
|
||||||
struct nasal_func// 112 bytes
|
struct nasal_func// 112 bytes
|
||||||
{
|
{
|
||||||
int32_t dynpara; // dynamic parameter name index in hash
|
uint32_t dynpara; // dynamic parameter index in hash. index 0 is reserved for 'me'.
|
||||||
uint32_t entry; // pc will set to entry-1 to call this function
|
uint32_t entry; // pc will set to entry-1 to call this function
|
||||||
std::vector<nasal_ref> local; // local scope with default value(nasal_ref)
|
std::vector<nasal_ref> local; // local scope with default value(nasal_ref)
|
||||||
std::vector<nasal_ref> upvalue; // closure
|
std::vector<nasal_ref> upvalue; // closure
|
||||||
std::unordered_map<std::string,int> key_table; // parameter name table
|
std::unordered_map<std::string,int> keys; // parameter name table
|
||||||
|
|
||||||
nasal_func():dynpara(-1){}
|
nasal_func():dynpara(0){}
|
||||||
void clear();
|
void clear();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -139,17 +139,17 @@ struct nasal_val// 16 bytes
|
||||||
/*functions of nasal_vec*/
|
/*functions of nasal_vec*/
|
||||||
nasal_ref nasal_vec::get_val(const int index)
|
nasal_ref nasal_vec::get_val(const int index)
|
||||||
{
|
{
|
||||||
int vec_size=elems.size();
|
int size=elems.size();
|
||||||
if(index<-vec_size || index>=vec_size)
|
if(index<-size || index>=size)
|
||||||
return {vm_none};
|
return {vm_none};
|
||||||
return elems[index>=0?index:index+vec_size];
|
return elems[index>=0?index:index+size];
|
||||||
}
|
}
|
||||||
nasal_ref* nasal_vec::get_mem(const int index)
|
nasal_ref* nasal_vec::get_mem(const int index)
|
||||||
{
|
{
|
||||||
int vec_size=elems.size();
|
int size=elems.size();
|
||||||
if(index<-vec_size || index>=vec_size)
|
if(index<-size || index>=size)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
return &elems[index>=0?index:index+vec_size];
|
return &elems[index>=0?index:index+size];
|
||||||
}
|
}
|
||||||
void nasal_vec::print()
|
void nasal_vec::print()
|
||||||
{
|
{
|
||||||
|
@ -212,15 +212,15 @@ nasal_ref* nasal_hash::get_mem(const std::string& key)
|
||||||
return &elems[key];
|
return &elems[key];
|
||||||
else if(elems.count("parents"))
|
else if(elems.count("parents"))
|
||||||
{
|
{
|
||||||
nasal_ref* mem_addr=nullptr;
|
nasal_ref* addr=nullptr;
|
||||||
nasal_ref val=elems["parents"];
|
nasal_ref val=elems["parents"];
|
||||||
if(val.type==vm_vec)
|
if(val.type==vm_vec)
|
||||||
for(auto& i:val.vec()->elems)
|
for(auto& i:val.vec()->elems)
|
||||||
{
|
{
|
||||||
if(i.type==vm_hash)
|
if(i.type==vm_hash)
|
||||||
mem_addr=i.hash()->get_mem(key);
|
addr=i.hash()->get_mem(key);
|
||||||
if(mem_addr)
|
if(addr)
|
||||||
return mem_addr;
|
return addr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
@ -265,9 +265,10 @@ void nasal_hash::print()
|
||||||
/*functions of nasal_func*/
|
/*functions of nasal_func*/
|
||||||
void nasal_func::clear()
|
void nasal_func::clear()
|
||||||
{
|
{
|
||||||
dynpara=-1;
|
dynpara=0;
|
||||||
local.clear();
|
local.clear();
|
||||||
key_table.clear();
|
upvalue.clear();
|
||||||
|
keys.clear();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -332,9 +333,9 @@ struct nasal_gc
|
||||||
nasal_ref zero; // reserved address of nasal_val,type vm_num, 0
|
nasal_ref zero; // reserved address of nasal_val,type vm_num, 0
|
||||||
nasal_ref one; // reserved address of nasal_val,type vm_num, 1
|
nasal_ref one; // reserved address of nasal_val,type vm_num, 1
|
||||||
nasal_ref nil; // reserved address of nasal_val,type vm_nil
|
nasal_ref nil; // reserved address of nasal_val,type vm_nil
|
||||||
nasal_ref val_stack[STACK_MAX_DEPTH+1];// 1 reserved to avoid stack overflow
|
nasal_ref stack[STACK_MAX_DEPTH+1];// 1 reserved to avoid stack overflow
|
||||||
nasal_ref* stack_top; // stack top
|
nasal_ref* top; // stack top
|
||||||
std::vector<nasal_ref> str_addrs; // reserved address for const vm_str
|
std::vector<nasal_ref> strs; // reserved address for const vm_str
|
||||||
std::vector<nasal_val*> memory; // gc memory
|
std::vector<nasal_val*> memory; // gc memory
|
||||||
std::queue<nasal_val*> free_list[vm_type_size]; // gc free list
|
std::queue<nasal_val*> free_list[vm_type_size]; // gc free list
|
||||||
std::vector<nasal_ref> local;
|
std::vector<nasal_ref> local;
|
||||||
|
@ -352,7 +353,7 @@ void nasal_gc::mark()
|
||||||
std::queue<nasal_ref> bfs;
|
std::queue<nasal_ref> bfs;
|
||||||
for(auto& i:local)
|
for(auto& i:local)
|
||||||
bfs.push(i);
|
bfs.push(i);
|
||||||
for(nasal_ref* i=val_stack;i<=stack_top;++i)
|
for(nasal_ref* i=stack;i<=top;++i)
|
||||||
bfs.push(*i);
|
bfs.push(*i);
|
||||||
while(!bfs.empty())
|
while(!bfs.empty())
|
||||||
{
|
{
|
||||||
|
@ -406,7 +407,7 @@ void nasal_gc::sweep()
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
void nasal_gc::init(const std::vector<std::string>& strs)
|
void nasal_gc::init(const std::vector<std::string>& s)
|
||||||
{
|
{
|
||||||
for(uint8_t i=vm_str;i<vm_type_size;++i)
|
for(uint8_t i=vm_str;i<vm_type_size;++i)
|
||||||
for(uint32_t j=0;j<increment[i];++j)
|
for(uint32_t j=0;j<increment[i];++j)
|
||||||
|
@ -416,18 +417,18 @@ void nasal_gc::init(const std::vector<std::string>& strs)
|
||||||
free_list[i].push(tmp);
|
free_list[i].push(tmp);
|
||||||
}
|
}
|
||||||
|
|
||||||
stack_top=val_stack; // set stack_top to val_stack
|
top=stack; // set top to stack
|
||||||
|
|
||||||
zero={vm_num,(double)0}; // init constant 0
|
zero={vm_num,(double)0}; // init constant 0
|
||||||
one ={vm_num,(double)1}; // init constant 1
|
one ={vm_num,(double)1}; // init constant 1
|
||||||
nil ={vm_nil,(double)0}; // init constant nil
|
nil ={vm_nil,(double)0}; // init constant nil
|
||||||
|
|
||||||
// init constant strings
|
// init constant strings
|
||||||
str_addrs.resize(strs.size());
|
strs.resize(s.size());
|
||||||
for(uint32_t i=0;i<strs.size();++i)
|
for(uint32_t i=0;i<strs.size();++i)
|
||||||
{
|
{
|
||||||
str_addrs[i]={vm_str,new nasal_val(vm_str)};
|
strs[i]={vm_str,new nasal_val(vm_str)};
|
||||||
*str_addrs[i].str()=strs[i];
|
*strs[i].str()=s[i];
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -440,9 +441,9 @@ void nasal_gc::clear()
|
||||||
while(!free_list[i].empty())
|
while(!free_list[i].empty())
|
||||||
free_list[i].pop();
|
free_list[i].pop();
|
||||||
local.clear();
|
local.clear();
|
||||||
for(auto& i:str_addrs)
|
for(auto& i:strs)
|
||||||
delete i.value.gcobj;
|
delete i.value.gcobj;
|
||||||
str_addrs.clear();
|
strs.clear();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
nasal_ref nasal_gc::alloc(uint8_t type)
|
nasal_ref nasal_gc::alloc(uint8_t type)
|
||||||
|
|
473
nasal_vm.h
473
nasal_vm.h
|
@ -5,10 +5,10 @@ class nasal_vm
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
/* reference from nasal_gc */
|
/* reference from nasal_gc */
|
||||||
nasal_ref*& stack_top;// stack top
|
nasal_ref*& top; // stack top
|
||||||
/* values of nasal_vm */
|
/* values of nasal_vm */
|
||||||
uint32_t pc; // program counter
|
uint32_t pc; // program counter
|
||||||
uint32_t newf_off; // used to load default parameter to a new function
|
uint32_t offset; // used to load default parameter to a new function
|
||||||
std::stack<uint32_t> ret; // ptr stack stores address for function to return
|
std::stack<uint32_t> ret; // ptr stack stores address for function to return
|
||||||
std::stack<nasal_func*> func_stk; // stack to store function,this is used when getting upvalues
|
std::stack<nasal_func*> func_stk; // stack to store function,this is used when getting upvalues
|
||||||
std::stack<int> counter; // iterator stack for forindex/foreach
|
std::stack<int> counter; // iterator stack for forindex/foreach
|
||||||
|
@ -61,25 +61,25 @@ private:
|
||||||
void opr_sub();
|
void opr_sub();
|
||||||
void opr_mul();
|
void opr_mul();
|
||||||
void opr_div();
|
void opr_div();
|
||||||
void opr_lnk();//
|
void opr_lnk();
|
||||||
void opr_addc();
|
void opr_addc();
|
||||||
void opr_subc();
|
void opr_subc();
|
||||||
void opr_mulc();
|
void opr_mulc();
|
||||||
void opr_divc();
|
void opr_divc();
|
||||||
void opr_lnkc();//
|
void opr_lnkc();
|
||||||
void opr_addeq();
|
void opr_addeq();
|
||||||
void opr_subeq();
|
void opr_subeq();
|
||||||
void opr_muleq();
|
void opr_muleq();
|
||||||
void opr_diveq();
|
void opr_diveq();
|
||||||
void opr_lnkeq();//
|
void opr_lnkeq();
|
||||||
void opr_addeqc();
|
void opr_addeqc();
|
||||||
void opr_subeqc();
|
void opr_subeqc();
|
||||||
void opr_muleqc();
|
void opr_muleqc();
|
||||||
void opr_diveqc();
|
void opr_diveqc();
|
||||||
void opr_lnkeqc();//
|
void opr_lnkeqc();
|
||||||
void opr_meq();
|
void opr_meq();
|
||||||
void opr_eq();//
|
void opr_eq();
|
||||||
void opr_neq();//
|
void opr_neq();
|
||||||
void opr_less();
|
void opr_less();
|
||||||
void opr_leq();
|
void opr_leq();
|
||||||
void opr_grt();
|
void opr_grt();
|
||||||
|
@ -99,24 +99,24 @@ private:
|
||||||
void opr_callg();
|
void opr_callg();
|
||||||
void opr_calll();
|
void opr_calll();
|
||||||
void opr_upval();
|
void opr_upval();
|
||||||
void opr_callv();//
|
void opr_callv();
|
||||||
void opr_callvi();//
|
void opr_callvi();
|
||||||
void opr_callh();//
|
void opr_callh();
|
||||||
void opr_callfv();//
|
void opr_callfv();
|
||||||
void opr_callfh();//
|
void opr_callfh();
|
||||||
void opr_callb();
|
void opr_callb();
|
||||||
void opr_slcbegin();
|
void opr_slcbegin();
|
||||||
void opr_slcend();
|
void opr_slcend();
|
||||||
void opr_slc();//
|
void opr_slc();
|
||||||
void opr_slc2();//
|
void opr_slc2();
|
||||||
void opr_mcallg();
|
void opr_mcallg();
|
||||||
void opr_mcalll();
|
void opr_mcalll();
|
||||||
void opr_mupval();
|
void opr_mupval();
|
||||||
void opr_mcallv();//
|
void opr_mcallv();
|
||||||
void opr_mcallh();
|
void opr_mcallh();
|
||||||
void opr_ret();
|
void opr_ret();
|
||||||
public:
|
public:
|
||||||
nasal_vm():stack_top(gc.stack_top){}
|
nasal_vm():top(gc.top){}
|
||||||
~nasal_vm(){clear();}
|
~nasal_vm(){clear();}
|
||||||
void run(
|
void run(
|
||||||
const nasal_codegen&,
|
const nasal_codegen&,
|
||||||
|
@ -147,17 +147,17 @@ void nasal_vm::clear()
|
||||||
}
|
}
|
||||||
void nasal_vm::valinfo(nasal_ref& val)
|
void nasal_vm::valinfo(nasal_ref& val)
|
||||||
{
|
{
|
||||||
const nasal_val* ptr=val.value.gcobj;
|
const nasal_val* p=val.value.gcobj;
|
||||||
switch(val.type)
|
switch(val.type)
|
||||||
{
|
{
|
||||||
case vm_none: printf("\tnull |\n");break;
|
case vm_none: printf("\tnull |\n");break;
|
||||||
case vm_nil: printf("\tnil |\n");break;
|
case vm_nil: printf("\tnil |\n");break;
|
||||||
case vm_num: printf("\tnum | %lf\n",val.num());break;
|
case vm_num: printf("\tnum | %lf\n",val.num());break;
|
||||||
case vm_str: printf("\tstr | <%p> %s\n",ptr,raw_string(*val.str()).c_str());break;
|
case vm_str: printf("\tstr | <%p> %s\n",p,raw_string(*val.str()).c_str());break;
|
||||||
case vm_func: printf("\tfunc | <%p> func{entry=0x%x}\n",ptr,val.func()->entry);break;
|
case vm_func: printf("\tfunc | <%p> func{entry=0x%x}\n",p,val.func()->entry);break;
|
||||||
case vm_vec: printf("\tvec | <%p> [%lu val]\n",ptr,val.vec()->elems.size());break;
|
case vm_vec: printf("\tvec | <%p> [%lu val]\n",p,val.vec()->elems.size());break;
|
||||||
case vm_hash: printf("\thash | <%p> {%lu member}\n",ptr,val.hash()->elems.size());break;
|
case vm_hash: printf("\thash | <%p> {%lu member}\n",p,val.hash()->elems.size());break;
|
||||||
case vm_obj: printf("\tobj | <%p>\n",ptr);break;
|
case vm_obj: printf("\tobj | <%p>\n",p);break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
void nasal_vm::bytecodeinfo(const uint32_t p)
|
void nasal_vm::bytecodeinfo(const uint32_t p)
|
||||||
|
@ -173,61 +173,63 @@ void nasal_vm::traceback()
|
||||||
// push pc to ret stack to store the position program crashed
|
// push pc to ret stack to store the position program crashed
|
||||||
ret.push(pc);
|
ret.push(pc);
|
||||||
printf("trace back:\n");
|
printf("trace back:\n");
|
||||||
uint32_t same_cnt=0,last_point=0xffffffff;
|
uint32_t same=0,last=0xffffffff;
|
||||||
for(uint32_t point=0;!ret.empty();last_point=point,ret.pop())
|
for(uint32_t point=0;!ret.empty();last=point,ret.pop())
|
||||||
{
|
{
|
||||||
point=ret.top();
|
point=ret.top();
|
||||||
if(point==last_point)
|
if(point==last)
|
||||||
{
|
{
|
||||||
++same_cnt;
|
++same;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if(same_cnt)
|
if(same)
|
||||||
{
|
{
|
||||||
printf("\t0x%.8x: %d same call(s) ...\n",last_point,same_cnt);
|
printf("\t0x%.8x: %d same call(s) ...\n",last,same);
|
||||||
same_cnt=0;
|
same=0;
|
||||||
}
|
}
|
||||||
bytecodeinfo(point);
|
bytecodeinfo(point);
|
||||||
}
|
}
|
||||||
if(same_cnt)
|
if(same)
|
||||||
printf("\t0x%.8x: %d same call(s) ...\n",last_point,same_cnt);
|
printf("\t0x%.8x: %d same call(s) ...\n",last,same);
|
||||||
}
|
}
|
||||||
void nasal_vm::stackinfo(const uint32_t limit)
|
void nasal_vm::stackinfo(const uint32_t limit=10)
|
||||||
{
|
{
|
||||||
printf("vm stack(limit %d):\n",limit);
|
printf("vm stack(limit %d):\n",limit);
|
||||||
uint32_t same_cnt=0;
|
uint32_t same=0,global_size=bytecode[0].num;
|
||||||
nasal_ref last_ptr={vm_none,0xffffffff};
|
nasal_ref last={vm_none,0xffffffff};
|
||||||
for(uint32_t i=0;i<limit && stack_top>=gc.val_stack;++i,--stack_top)
|
for(uint32_t i=0;i<limit && top>=gc.stack+global_size;++i,--top)
|
||||||
{
|
{
|
||||||
if(stack_top[0]==last_ptr)
|
if(top[0]==last)
|
||||||
{
|
{
|
||||||
++same_cnt;
|
++same;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if(same_cnt)
|
if(same)
|
||||||
{
|
{
|
||||||
printf("\t... | %d same value(s)\n",same_cnt);
|
printf("\t... | %d same value(s)\n",same);
|
||||||
same_cnt=0;
|
same=0;
|
||||||
}
|
}
|
||||||
last_ptr=stack_top[0];
|
last=top[0];
|
||||||
valinfo(stack_top[0]);
|
valinfo(top[0]);
|
||||||
}
|
}
|
||||||
if(same_cnt)
|
if(same)
|
||||||
printf("\t... | %d same value(s)\n",same_cnt);
|
printf("\t... | %d same value(s)\n",same);
|
||||||
}
|
}
|
||||||
void nasal_vm::detail()
|
void nasal_vm::detail()
|
||||||
{
|
{
|
||||||
printf("mcall address: %p\n",mem_addr);
|
printf("mcall address: %p\n",mem_addr);
|
||||||
printf("global value:\n");
|
if(bytecode[0].num)// bytecode[0] is op_intg
|
||||||
// bytecode[0] is op_intg
|
{
|
||||||
|
printf("global:\n");
|
||||||
for(uint32_t i=0;i<bytecode[0].num;++i)
|
for(uint32_t i=0;i<bytecode[0].num;++i)
|
||||||
{
|
{
|
||||||
printf("[%d]",i);
|
printf("[%d]",i);
|
||||||
valinfo(gc.val_stack[i]);
|
valinfo(gc.stack[i]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if(!gc.local.empty())
|
if(!gc.local.empty())
|
||||||
{
|
{
|
||||||
printf("local value:\n");
|
printf("local:\n");
|
||||||
auto& vec=gc.local.back().vec()->elems;
|
auto& vec=gc.local.back().vec()->elems;
|
||||||
for(uint32_t i=0;i<vec.size();++i)
|
for(uint32_t i=0;i<vec.size();++i)
|
||||||
{
|
{
|
||||||
|
@ -235,6 +237,20 @@ void nasal_vm::detail()
|
||||||
valinfo(vec[i]);
|
valinfo(vec[i]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if(!func_stk.empty() && !func_stk.top()->upvalue.empty())
|
||||||
|
{
|
||||||
|
printf("upvalue:\n");
|
||||||
|
auto& upval=func_stk.top()->upvalue;
|
||||||
|
for(uint32_t i=0;i<upval.size();++i)
|
||||||
|
{
|
||||||
|
auto& vec=upval[i].vec()->elems;
|
||||||
|
for(uint32_t j=0;j<vec.size();++j)
|
||||||
|
{
|
||||||
|
printf("[%d][%d]",i,j);
|
||||||
|
valinfo(vec[j]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
void nasal_vm::opcallsort(const uint64_t* arr)
|
void nasal_vm::opcallsort(const uint64_t* arr)
|
||||||
{
|
{
|
||||||
|
@ -259,7 +275,7 @@ void nasal_vm::die(std::string str)
|
||||||
{
|
{
|
||||||
printf("[vm] %s\n",str.c_str());
|
printf("[vm] %s\n",str.c_str());
|
||||||
traceback();
|
traceback();
|
||||||
stackinfo(10);
|
stackinfo();
|
||||||
if(detail_info)
|
if(detail_info)
|
||||||
detail();
|
detail();
|
||||||
std::exit(1);
|
std::exit(1);
|
||||||
|
@ -283,111 +299,108 @@ inline void nasal_vm::opr_intg()
|
||||||
{
|
{
|
||||||
// global values store on stack
|
// global values store on stack
|
||||||
for(uint32_t i=0;i<imm[pc];++i)
|
for(uint32_t i=0;i<imm[pc];++i)
|
||||||
(stack_top++)[0].type=vm_nil;
|
(top++)[0].type=vm_nil;
|
||||||
--stack_top;// point to the top
|
--top;// point to the top
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_intl()
|
inline void nasal_vm::opr_intl()
|
||||||
{
|
{
|
||||||
stack_top[0].func()->local.resize(imm[pc],gc.nil);
|
top[0].func()->local.resize(imm[pc],gc.nil);
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_loadg()
|
inline void nasal_vm::opr_loadg()
|
||||||
{
|
{
|
||||||
gc.val_stack[imm[pc]]=(stack_top--)[0];
|
gc.stack[imm[pc]]=(top--)[0];
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_loadl()
|
inline void nasal_vm::opr_loadl()
|
||||||
{
|
{
|
||||||
gc.local.back().vec()->elems[imm[pc]]=(stack_top--)[0];
|
gc.local.back().vec()->elems[imm[pc]]=(top--)[0];
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_loadu()
|
inline void nasal_vm::opr_loadu()
|
||||||
{
|
{
|
||||||
func_stk.top()->upvalue[(imm[pc]>>16)&0xffff].vec()->elems[imm[pc]&0xffff]=(stack_top--)[0];
|
func_stk.top()->upvalue[(imm[pc]>>16)&0xffff].vec()->elems[imm[pc]&0xffff]=(top--)[0];
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_pnum()
|
inline void nasal_vm::opr_pnum()
|
||||||
{
|
{
|
||||||
(++stack_top)[0]={vm_num,num_table[imm[pc]]};
|
(++top)[0]={vm_num,num_table[imm[pc]]};
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_pone()
|
inline void nasal_vm::opr_pone()
|
||||||
{
|
{
|
||||||
(++stack_top)[0]={vm_num,(double)1};
|
(++top)[0]={vm_num,(double)1};
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_pzero()
|
inline void nasal_vm::opr_pzero()
|
||||||
{
|
{
|
||||||
(++stack_top)[0]={vm_num,(double)0};
|
(++top)[0]={vm_num,(double)0};
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_pnil()
|
inline void nasal_vm::opr_pnil()
|
||||||
{
|
{
|
||||||
(++stack_top)[0].type=vm_nil;
|
(++top)[0].type=vm_nil;
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_pstr()
|
inline void nasal_vm::opr_pstr()
|
||||||
{
|
{
|
||||||
(++stack_top)[0]=gc.str_addrs[imm[pc]];
|
(++top)[0]=gc.strs[imm[pc]];
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_newv()
|
inline void nasal_vm::opr_newv()
|
||||||
{
|
{
|
||||||
nasal_ref vec_addr=gc.alloc(vm_vec);
|
nasal_ref newv=gc.alloc(vm_vec);
|
||||||
nasal_ref* begin=stack_top-imm[pc]+1;
|
auto& vec=newv.vec()->elems;// top-imm[pc] stores the vector
|
||||||
auto& vec=vec_addr.vec()->elems;// stack_top-imm[pc] stores the vector
|
|
||||||
vec.resize(imm[pc]);
|
vec.resize(imm[pc]);
|
||||||
|
top-=imm[pc]-1;
|
||||||
for(uint32_t i=0;i<imm[pc];++i)
|
for(uint32_t i=0;i<imm[pc];++i)
|
||||||
vec[i]=begin[i];
|
vec[i]=top[i];
|
||||||
begin[0]=vec_addr;
|
top[0]=newv;
|
||||||
stack_top=begin;
|
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_newh()
|
inline void nasal_vm::opr_newh()
|
||||||
{
|
{
|
||||||
(++stack_top)[0]=gc.alloc(vm_hash);
|
(++top)[0]=gc.alloc(vm_hash);
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_newf()
|
inline void nasal_vm::opr_newf()
|
||||||
{
|
{
|
||||||
newf_off=1;
|
offset=1;
|
||||||
(++stack_top)[0]=gc.alloc(vm_func);
|
(++top)[0]=gc.alloc(vm_func);
|
||||||
stack_top[0].func()->entry=imm[pc];
|
top[0].func()->entry=imm[pc];
|
||||||
if(!gc.local.empty())
|
if(!gc.local.empty())
|
||||||
{
|
{
|
||||||
stack_top[0].func()->upvalue=func_stk.top()->upvalue;
|
top[0].func()->upvalue=func_stk.top()->upvalue;
|
||||||
stack_top[0].func()->upvalue.push_back(gc.local.back());
|
top[0].func()->upvalue.push_back(gc.local.back());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_happ()
|
inline void nasal_vm::opr_happ()
|
||||||
{
|
{
|
||||||
stack_top[-1].hash()->elems[str_table[imm[pc]]]=stack_top[0];
|
top[-1].hash()->elems[str_table[imm[pc]]]=top[0];
|
||||||
--stack_top;
|
--top;
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_para()
|
inline void nasal_vm::opr_para()
|
||||||
{
|
{
|
||||||
nasal_func* func=stack_top[0].func();
|
nasal_func* func=top[0].func();
|
||||||
size_t size=func->key_table.size();
|
size_t size=func->keys.size();
|
||||||
func->key_table[str_table[imm[pc]]]=size;
|
func->keys[str_table[imm[pc]]]=size;
|
||||||
func->local[newf_off]={vm_none};
|
func->local[offset++]={vm_none};
|
||||||
++newf_off;
|
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_defpara()
|
inline void nasal_vm::opr_defpara()
|
||||||
{
|
{
|
||||||
nasal_ref def_val=stack_top[0];
|
nasal_ref val=top[0];
|
||||||
nasal_func* func=(--stack_top)[0].func();
|
nasal_func* func=(--top)[0].func();
|
||||||
size_t size=func->key_table.size();
|
size_t size=func->keys.size();
|
||||||
func->key_table[str_table[imm[pc]]]=size;
|
func->keys[str_table[imm[pc]]]=size;
|
||||||
func->local[newf_off]=def_val;
|
func->local[offset++]=val;
|
||||||
++newf_off;
|
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_dynpara()
|
inline void nasal_vm::opr_dynpara()
|
||||||
{
|
{
|
||||||
stack_top[0].func()->dynpara=imm[pc];
|
top[0].func()->dynpara=imm[pc];
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_unot()
|
inline void nasal_vm::opr_unot()
|
||||||
{
|
{
|
||||||
nasal_ref val=stack_top[0];
|
nasal_ref val=top[0];
|
||||||
switch(val.type)
|
switch(val.type)
|
||||||
{
|
{
|
||||||
case vm_nil:stack_top[0]=gc.zero;break;
|
case vm_nil:top[0]=gc.zero;break;
|
||||||
case vm_num:stack_top[0]=val.num()?gc.zero:gc.one;break;
|
case vm_num:top[0]=val.num()?gc.zero:gc.one;break;
|
||||||
case vm_str:
|
case vm_str:
|
||||||
{
|
{
|
||||||
double num=str2num(val.str()->c_str());
|
double num=str2num(val.str()->c_str());
|
||||||
if(std::isnan(num))
|
if(std::isnan(num))
|
||||||
stack_top[0]=val.str()->empty()?gc.one:gc.zero;
|
top[0]=val.str()->empty()?gc.one:gc.zero;
|
||||||
else
|
else
|
||||||
stack_top[0]=num?gc.zero:gc.one;
|
top[0]=num?gc.zero:gc.one;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
default:die("unot: incorrect value type");break;
|
default:die("unot: incorrect value type");break;
|
||||||
|
@ -395,12 +408,12 @@ inline void nasal_vm::opr_unot()
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_usub()
|
inline void nasal_vm::opr_usub()
|
||||||
{
|
{
|
||||||
stack_top[0]={vm_num,-stack_top[0].to_number()};
|
top[0]={vm_num,-top[0].to_number()};
|
||||||
}
|
}
|
||||||
|
|
||||||
#define op_calc(type)\
|
#define op_calc(type)\
|
||||||
nasal_ref new_val(vm_num,stack_top[-1].to_number() type stack_top[0].to_number());\
|
nasal_ref val(vm_num,top[-1].to_number() type top[0].to_number());\
|
||||||
(--stack_top)[0]=new_val;
|
(--top)[0]=val;
|
||||||
|
|
||||||
inline void nasal_vm::opr_add(){op_calc(+);}
|
inline void nasal_vm::opr_add(){op_calc(+);}
|
||||||
inline void nasal_vm::opr_sub(){op_calc(-);}
|
inline void nasal_vm::opr_sub(){op_calc(-);}
|
||||||
|
@ -408,14 +421,14 @@ inline void nasal_vm::opr_mul(){op_calc(*);}
|
||||||
inline void nasal_vm::opr_div(){op_calc(/);}
|
inline void nasal_vm::opr_div(){op_calc(/);}
|
||||||
inline void nasal_vm::opr_lnk()
|
inline void nasal_vm::opr_lnk()
|
||||||
{
|
{
|
||||||
nasal_ref new_val=gc.alloc(vm_str);
|
nasal_ref val=gc.alloc(vm_str);
|
||||||
*new_val.str()=stack_top[-1].to_string()+stack_top[0].to_string();
|
*val.str()=top[-1].to_string()+top[0].to_string();
|
||||||
(--stack_top)[0]=new_val;
|
(--top)[0]=val;
|
||||||
}
|
}
|
||||||
|
|
||||||
#define op_calc_const(type)\
|
#define op_calc_const(type)\
|
||||||
nasal_ref new_val(vm_num,stack_top[0].to_number() type num_table[imm[pc]]);\
|
nasal_ref val(vm_num,top[0].to_number() type num_table[imm[pc]]);\
|
||||||
stack_top[0]=new_val;
|
top[0]=val;
|
||||||
|
|
||||||
inline void nasal_vm::opr_addc(){op_calc_const(+);}
|
inline void nasal_vm::opr_addc(){op_calc_const(+);}
|
||||||
inline void nasal_vm::opr_subc(){op_calc_const(-);}
|
inline void nasal_vm::opr_subc(){op_calc_const(-);}
|
||||||
|
@ -423,14 +436,14 @@ inline void nasal_vm::opr_mulc(){op_calc_const(*);}
|
||||||
inline void nasal_vm::opr_divc(){op_calc_const(/);}
|
inline void nasal_vm::opr_divc(){op_calc_const(/);}
|
||||||
inline void nasal_vm::opr_lnkc()
|
inline void nasal_vm::opr_lnkc()
|
||||||
{
|
{
|
||||||
nasal_ref new_val=gc.alloc(vm_str);
|
nasal_ref val=gc.alloc(vm_str);
|
||||||
*new_val.str()=stack_top[0].to_string()+str_table[imm[pc]];
|
*val.str()=top[0].to_string()+str_table[imm[pc]];
|
||||||
stack_top[0]=new_val;
|
top[0]=val;
|
||||||
}
|
}
|
||||||
|
|
||||||
#define op_calc_eq(type)\
|
#define op_calc_eq(type)\
|
||||||
nasal_ref new_val(vm_num,mem_addr[0].to_number() type stack_top[-1].to_number());\
|
nasal_ref val(vm_num,mem_addr[0].to_number() type top[-1].to_number());\
|
||||||
(--stack_top)[0]=mem_addr[0]=new_val;
|
(--top)[0]=mem_addr[0]=val;
|
||||||
|
|
||||||
inline void nasal_vm::opr_addeq(){op_calc_eq(+);}
|
inline void nasal_vm::opr_addeq(){op_calc_eq(+);}
|
||||||
inline void nasal_vm::opr_subeq(){op_calc_eq(-);}
|
inline void nasal_vm::opr_subeq(){op_calc_eq(-);}
|
||||||
|
@ -438,14 +451,14 @@ inline void nasal_vm::opr_muleq(){op_calc_eq(*);}
|
||||||
inline void nasal_vm::opr_diveq(){op_calc_eq(/);}
|
inline void nasal_vm::opr_diveq(){op_calc_eq(/);}
|
||||||
inline void nasal_vm::opr_lnkeq()
|
inline void nasal_vm::opr_lnkeq()
|
||||||
{
|
{
|
||||||
nasal_ref new_val=gc.alloc(vm_str);
|
nasal_ref val=gc.alloc(vm_str);
|
||||||
*new_val.str()=mem_addr[0].to_string()+stack_top[-1].to_string();
|
*val.str()=mem_addr[0].to_string()+top[-1].to_string();
|
||||||
(--stack_top)[0]=mem_addr[0]=new_val;
|
(--top)[0]=mem_addr[0]=val;
|
||||||
}
|
}
|
||||||
|
|
||||||
#define op_calc_eq_const(type)\
|
#define op_calc_eq_const(type)\
|
||||||
nasal_ref new_val(vm_num,mem_addr[0].to_number() type num_table[imm[pc]]);\
|
nasal_ref val(vm_num,mem_addr[0].to_number() type num_table[imm[pc]]);\
|
||||||
stack_top[0]=mem_addr[0]=new_val;
|
top[0]=mem_addr[0]=val;
|
||||||
|
|
||||||
inline void nasal_vm::opr_addeqc(){op_calc_eq_const(+);}
|
inline void nasal_vm::opr_addeqc(){op_calc_eq_const(+);}
|
||||||
inline void nasal_vm::opr_subeqc(){op_calc_eq_const(-);}
|
inline void nasal_vm::opr_subeqc(){op_calc_eq_const(-);}
|
||||||
|
@ -453,49 +466,49 @@ inline void nasal_vm::opr_muleqc(){op_calc_eq_const(*);}
|
||||||
inline void nasal_vm::opr_diveqc(){op_calc_eq_const(/);}
|
inline void nasal_vm::opr_diveqc(){op_calc_eq_const(/);}
|
||||||
inline void nasal_vm::opr_lnkeqc()
|
inline void nasal_vm::opr_lnkeqc()
|
||||||
{
|
{
|
||||||
nasal_ref new_val=gc.alloc(vm_str);
|
nasal_ref val=gc.alloc(vm_str);
|
||||||
*new_val.str()=mem_addr[0].to_string()+str_table[imm[pc]];
|
*val.str()=mem_addr[0].to_string()+str_table[imm[pc]];
|
||||||
stack_top[0]=mem_addr[0]=new_val;
|
top[0]=mem_addr[0]=val;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline void nasal_vm::opr_meq()
|
inline void nasal_vm::opr_meq()
|
||||||
{
|
{
|
||||||
mem_addr[0]=(--stack_top)[0];
|
mem_addr[0]=(--top)[0];
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_eq()
|
inline void nasal_vm::opr_eq()
|
||||||
{
|
{
|
||||||
nasal_ref val2=stack_top[0];
|
nasal_ref val2=top[0];
|
||||||
nasal_ref val1=(--stack_top)[0];
|
nasal_ref val1=(--top)[0];
|
||||||
uint8_t a_type=val1.type;
|
uint8_t a_type=val1.type;
|
||||||
uint8_t b_type=val2.type;
|
uint8_t b_type=val2.type;
|
||||||
if(a_type==vm_nil && b_type==vm_nil)
|
if(a_type==vm_nil && b_type==vm_nil)
|
||||||
stack_top[0]=gc.one;
|
top[0]=gc.one;
|
||||||
else if(a_type==vm_str && b_type==vm_str)
|
else if(a_type==vm_str && b_type==vm_str)
|
||||||
stack_top[0]=(*val1.str()==*val2.str())?gc.one:gc.zero;
|
top[0]=(*val1.str()==*val2.str())?gc.one:gc.zero;
|
||||||
else if(a_type==vm_num || b_type==vm_num)
|
else if(a_type==vm_num || b_type==vm_num)
|
||||||
stack_top[0]=(val1.to_number()==val2.to_number())?gc.one:gc.zero;
|
top[0]=(val1.to_number()==val2.to_number())?gc.one:gc.zero;
|
||||||
else
|
else
|
||||||
stack_top[0]=(val1==val2)?gc.one:gc.zero;
|
top[0]=(val1==val2)?gc.one:gc.zero;
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_neq()
|
inline void nasal_vm::opr_neq()
|
||||||
{
|
{
|
||||||
nasal_ref val2=stack_top[0];
|
nasal_ref val2=top[0];
|
||||||
nasal_ref val1=(--stack_top)[0];
|
nasal_ref val1=(--top)[0];
|
||||||
uint8_t a_type=val1.type;
|
uint8_t a_type=val1.type;
|
||||||
uint8_t b_type=val2.type;
|
uint8_t b_type=val2.type;
|
||||||
if(a_type==vm_nil && b_type==vm_nil)
|
if(a_type==vm_nil && b_type==vm_nil)
|
||||||
stack_top[0]=gc.zero;
|
top[0]=gc.zero;
|
||||||
else if(a_type==vm_str && b_type==vm_str)
|
else if(a_type==vm_str && b_type==vm_str)
|
||||||
stack_top[0]=(*val1.str()!=*val2.str())?gc.one:gc.zero;
|
top[0]=(*val1.str()!=*val2.str())?gc.one:gc.zero;
|
||||||
else if(a_type==vm_num || b_type==vm_num)
|
else if(a_type==vm_num || b_type==vm_num)
|
||||||
stack_top[0]=(val1.to_number()!=val2.to_number())?gc.one:gc.zero;
|
top[0]=(val1.to_number()!=val2.to_number())?gc.one:gc.zero;
|
||||||
else
|
else
|
||||||
stack_top[0]=(val1!=val2)?gc.one:gc.zero;
|
top[0]=(val1!=val2)?gc.one:gc.zero;
|
||||||
}
|
}
|
||||||
|
|
||||||
#define op_cmp(type)\
|
#define op_cmp(type)\
|
||||||
--stack_top;\
|
--top;\
|
||||||
stack_top[0]=(stack_top[0].to_number() type stack_top[1].to_number())?gc.one:gc.zero;
|
top[0]=(top[0].to_number() type top[1].to_number())?gc.one:gc.zero;
|
||||||
|
|
||||||
inline void nasal_vm::opr_less(){op_cmp(<);}
|
inline void nasal_vm::opr_less(){op_cmp(<);}
|
||||||
inline void nasal_vm::opr_leq(){op_cmp(<=);}
|
inline void nasal_vm::opr_leq(){op_cmp(<=);}
|
||||||
|
@ -503,7 +516,7 @@ inline void nasal_vm::opr_grt(){op_cmp(>);}
|
||||||
inline void nasal_vm::opr_geq(){op_cmp(>=);}
|
inline void nasal_vm::opr_geq(){op_cmp(>=);}
|
||||||
|
|
||||||
#define op_cmp_const(type)\
|
#define op_cmp_const(type)\
|
||||||
stack_top[0]=(stack_top[0].to_number() type num_table[imm[pc]])?gc.one:gc.zero;
|
top[0]=(top[0].to_number() type num_table[imm[pc]])?gc.one:gc.zero;
|
||||||
|
|
||||||
inline void nasal_vm::opr_lessc(){op_cmp_const(<);}
|
inline void nasal_vm::opr_lessc(){op_cmp_const(<);}
|
||||||
inline void nasal_vm::opr_leqc(){op_cmp_const(<=);}
|
inline void nasal_vm::opr_leqc(){op_cmp_const(<=);}
|
||||||
|
@ -512,7 +525,7 @@ inline void nasal_vm::opr_geqc(){op_cmp_const(>=);}
|
||||||
|
|
||||||
inline void nasal_vm::opr_pop()
|
inline void nasal_vm::opr_pop()
|
||||||
{
|
{
|
||||||
--stack_top;
|
--top;
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_jmp()
|
inline void nasal_vm::opr_jmp()
|
||||||
{
|
{
|
||||||
|
@ -520,19 +533,19 @@ inline void nasal_vm::opr_jmp()
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_jt()
|
inline void nasal_vm::opr_jt()
|
||||||
{
|
{
|
||||||
if(condition(stack_top[0]))
|
if(condition(top[0]))
|
||||||
pc=imm[pc]-1;
|
pc=imm[pc]-1;
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_jf()
|
inline void nasal_vm::opr_jf()
|
||||||
{
|
{
|
||||||
if(!condition(stack_top[0]))
|
if(!condition(top[0]))
|
||||||
pc=imm[pc]-1;
|
pc=imm[pc]-1;
|
||||||
--stack_top;
|
--top;
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_counter()
|
inline void nasal_vm::opr_counter()
|
||||||
{
|
{
|
||||||
counter.push(-1);
|
counter.push(-1);
|
||||||
if(stack_top[0].type!=vm_vec)
|
if(top[0].type!=vm_vec)
|
||||||
die("cnt: must use vector in forindex/foreach");
|
die("cnt: must use vector in forindex/foreach");
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_cntpop()
|
inline void nasal_vm::opr_cntpop()
|
||||||
|
@ -541,54 +554,54 @@ inline void nasal_vm::opr_cntpop()
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_findex()
|
inline void nasal_vm::opr_findex()
|
||||||
{
|
{
|
||||||
if(++counter.top()>=stack_top[0].vec()->elems.size())
|
if(++counter.top()>=top[0].vec()->elems.size())
|
||||||
{
|
{
|
||||||
pc=imm[pc]-1;
|
pc=imm[pc]-1;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
(++stack_top)[0]={vm_num,static_cast<double>(counter.top())};
|
(++top)[0]={vm_num,static_cast<double>(counter.top())};
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_feach()
|
inline void nasal_vm::opr_feach()
|
||||||
{
|
{
|
||||||
std::vector<nasal_ref>& ref=stack_top[0].vec()->elems;
|
std::vector<nasal_ref>& ref=top[0].vec()->elems;
|
||||||
if(++counter.top()>=ref.size())
|
if(++counter.top()>=ref.size())
|
||||||
{
|
{
|
||||||
pc=imm[pc]-1;
|
pc=imm[pc]-1;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
(++stack_top)[0]=ref[counter.top()];
|
(++top)[0]=ref[counter.top()];
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_callg()
|
inline void nasal_vm::opr_callg()
|
||||||
{
|
{
|
||||||
(++stack_top)[0]=gc.val_stack[imm[pc]];
|
(++top)[0]=gc.stack[imm[pc]];
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_calll()
|
inline void nasal_vm::opr_calll()
|
||||||
{
|
{
|
||||||
(++stack_top)[0]=gc.local.back().vec()->elems[imm[pc]];
|
(++top)[0]=gc.local.back().vec()->elems[imm[pc]];
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_upval()
|
inline void nasal_vm::opr_upval()
|
||||||
{
|
{
|
||||||
(++stack_top)[0]=func_stk.top()->upvalue[(imm[pc]>>16)&0xffff].vec()->elems[imm[pc]&0xffff];
|
(++top)[0]=func_stk.top()->upvalue[(imm[pc]>>16)&0xffff].vec()->elems[imm[pc]&0xffff];
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_callv()
|
inline void nasal_vm::opr_callv()
|
||||||
{
|
{
|
||||||
nasal_ref val=stack_top[0];
|
nasal_ref val=top[0];
|
||||||
nasal_ref vec=(--stack_top)[0];
|
nasal_ref vec=(--top)[0];
|
||||||
if(vec.type==vm_vec)
|
if(vec.type==vm_vec)
|
||||||
{
|
{
|
||||||
stack_top[0]=vec.vec()->get_val(val.to_number());
|
top[0]=vec.vec()->get_val(val.to_number());
|
||||||
if(stack_top[0].type==vm_none)
|
if(top[0].type==vm_none)
|
||||||
die("callv: index out of range:"+std::to_string(val.to_number()));
|
die("callv: index out of range:"+std::to_string(val.to_number()));
|
||||||
}
|
}
|
||||||
else if(vec.type==vm_hash)
|
else if(vec.type==vm_hash)
|
||||||
{
|
{
|
||||||
if(val.type!=vm_str)
|
if(val.type!=vm_str)
|
||||||
die("callv: must use string as the key");
|
die("callv: must use string as the key");
|
||||||
stack_top[0]=vec.hash()->get_val(*val.value.gcobj->ptr.str);
|
top[0]=vec.hash()->get_val(*val.value.gcobj->ptr.str);
|
||||||
if(stack_top[0].type==vm_none)
|
if(top[0].type==vm_none)
|
||||||
die("callv: cannot find member \""+*val.str()+"\" of this hash");
|
die("callv: cannot find member \""+*val.str()+"\" of this hash");
|
||||||
if(stack_top[0].type==vm_func)
|
if(top[0].type==vm_func)
|
||||||
stack_top[0].func()->local[0]=val;// me
|
top[0].func()->local[0]=val;// me
|
||||||
}
|
}
|
||||||
else if(vec.type==vm_str)
|
else if(vec.type==vm_str)
|
||||||
{
|
{
|
||||||
|
@ -597,194 +610,190 @@ inline void nasal_vm::opr_callv()
|
||||||
int str_size=str.length();
|
int str_size=str.length();
|
||||||
if(num<-str_size || num>=str_size)
|
if(num<-str_size || num>=str_size)
|
||||||
die("callv: index out of range:"+std::to_string(val.to_number()));
|
die("callv: index out of range:"+std::to_string(val.to_number()));
|
||||||
stack_top[0]={vm_num,static_cast<double>(str[num>=0? num:num+str_size])};
|
top[0]={vm_num,static_cast<double>(str[num>=0? num:num+str_size])};
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
die("callv: must call a vector/hash/string");
|
die("callv: must call a vector/hash/string");
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_callvi()
|
inline void nasal_vm::opr_callvi()
|
||||||
{
|
{
|
||||||
nasal_ref val=stack_top[0];
|
nasal_ref val=top[0];
|
||||||
if(val.type!=vm_vec)
|
if(val.type!=vm_vec)
|
||||||
die("callvi: must use a vector");
|
die("callvi: must use a vector");
|
||||||
|
|
||||||
// cannot use operator[],because this may cause overflow
|
// cannot use operator[],because this may cause overflow
|
||||||
(++stack_top)[0]=val.vec()->get_val(imm[pc]);
|
(++top)[0]=val.vec()->get_val(imm[pc]);
|
||||||
if(stack_top[0].type==vm_none)
|
if(top[0].type==vm_none)
|
||||||
die("callvi: index out of range:"+std::to_string(imm[pc]));
|
die("callvi: index out of range:"+std::to_string(imm[pc]));
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_callh()
|
inline void nasal_vm::opr_callh()
|
||||||
{
|
{
|
||||||
nasal_ref val=stack_top[0];
|
nasal_ref val=top[0];
|
||||||
if(val.type!=vm_hash)
|
if(val.type!=vm_hash)
|
||||||
die("callh: must call a hash");
|
die("callh: must call a hash");
|
||||||
|
|
||||||
stack_top[0]=val.hash()->get_val(str_table[imm[pc]]);
|
top[0]=val.hash()->get_val(str_table[imm[pc]]);
|
||||||
if(stack_top[0].type==vm_none)
|
if(top[0].type==vm_none)
|
||||||
die("callh: member \""+str_table[imm[pc]]+"\" does not exist");
|
die("callh: member \""+str_table[imm[pc]]+"\" does not exist");
|
||||||
|
|
||||||
if(stack_top[0].type==vm_func)
|
if(top[0].type==vm_func)
|
||||||
stack_top[0].func()->local[0]=val;// me
|
top[0].func()->local[0]=val;// me
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_callfv()
|
inline void nasal_vm::opr_callfv()
|
||||||
{
|
{
|
||||||
// get parameter list and function value
|
size_t args_size=imm[pc];
|
||||||
uint32_t args_size=imm[pc];
|
nasal_ref* args=top-args_size+1;
|
||||||
nasal_ref* vec=stack_top-args_size+1;
|
if(args[-1].type!=vm_func)
|
||||||
nasal_ref func_addr=vec[-1];
|
|
||||||
if(func_addr.type!=vm_func)
|
|
||||||
die("callfv: must call a function");
|
die("callfv: must call a function");
|
||||||
// push new local scope
|
// push new local scope
|
||||||
func_stk.push(func_addr.func());
|
func_stk.push(args[-1].func());
|
||||||
auto& ref_func=*func_addr.func();
|
auto& func=*args[-1].func();
|
||||||
|
|
||||||
gc.local.push_back(gc.alloc(vm_vec));
|
gc.local.push_back(gc.alloc(vm_vec));
|
||||||
gc.local.back().vec()->elems=ref_func.local;
|
gc.local.back().vec()->elems=func.local;
|
||||||
// load parameters
|
// load parameters
|
||||||
auto& ref_closure=gc.local.back().vec()->elems;
|
auto& closure=gc.local.back().vec()->elems;
|
||||||
|
|
||||||
uint32_t para_size=ref_func.key_table.size();
|
size_t para_size=func.keys.size();
|
||||||
// load arguments
|
// load arguments
|
||||||
// if the first default value is not vm_none,then values after it are not nullptr
|
// if the first default value is not vm_none,then values after it are not nullptr
|
||||||
if(args_size<para_size && ref_func.local[args_size+1/*1 is reserved for 'me'*/].type==vm_none)
|
if(args_size<para_size && func.local[args_size+1/*1 is reserved for 'me'*/].type==vm_none)
|
||||||
die("callfv: lack argument(s)");
|
die("callfv: lack argument(s)");
|
||||||
// if args_size>para_size,for 0 to args_size will cause corruption
|
// if args_size>para_size,for 0 to args_size will cause corruption
|
||||||
uint32_t min_size=std::min(para_size,args_size);
|
size_t min_size=std::min(para_size,args_size);
|
||||||
for(uint32_t i=0;i<min_size;++i)
|
for(uint32_t i=0;i<min_size;++i)
|
||||||
ref_closure[i+1]=vec[i];
|
closure[i+1]=args[i];
|
||||||
// load dynamic argument if args_size>=para_size
|
// load dynamic argument if args_size>=para_size
|
||||||
if(ref_func.dynpara>=0)
|
if(func.dynpara)
|
||||||
{
|
{
|
||||||
nasal_ref vec_addr=gc.alloc(vm_vec);
|
nasal_ref vec=gc.alloc(vm_vec);
|
||||||
for(uint32_t i=para_size;i<args_size;++i)
|
for(uint32_t i=para_size;i<args_size;++i)
|
||||||
vec_addr.vec()->elems.push_back(vec[i]);
|
vec.vec()->elems.push_back(args[i]);
|
||||||
ref_closure.back()=vec_addr;
|
closure.back()=vec;
|
||||||
}
|
}
|
||||||
|
|
||||||
stack_top-=args_size;// pop arguments
|
top-=args_size;// pop arguments
|
||||||
ret.push(pc);
|
ret.push(pc);
|
||||||
pc=ref_func.entry-1;
|
pc=func.entry-1;
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_callfh()
|
inline void nasal_vm::opr_callfh()
|
||||||
{
|
{
|
||||||
// get parameter list and function value
|
// get parameter list and function value
|
||||||
auto& ref_hash=stack_top[0].hash()->elems;
|
auto& hash=top[0].hash()->elems;
|
||||||
nasal_ref func_addr=stack_top[-1];
|
if(top[-1].type!=vm_func)
|
||||||
if(func_addr.type!=vm_func)
|
|
||||||
die("callfh: must call a function");
|
die("callfh: must call a function");
|
||||||
// push new local scope
|
// push new local scope
|
||||||
func_stk.push(func_addr.func());
|
func_stk.push(top[-1].func());
|
||||||
auto& ref_func=*func_addr.func();
|
auto& func=*top[-1].func();
|
||||||
gc.local.push_back(gc.alloc(vm_vec));
|
gc.local.push_back(gc.alloc(vm_vec));
|
||||||
gc.local.back().vec()->elems=ref_func.local;
|
gc.local.back().vec()->elems=func.local;
|
||||||
// load parameters
|
// load parameters
|
||||||
|
auto& closure=gc.local.back().vec()->elems;
|
||||||
auto& ref_closure=gc.local.back().vec()->elems;
|
if(func.dynpara)
|
||||||
if(ref_func.dynpara>=0)
|
|
||||||
die("callfh: special call cannot use dynamic argument");
|
die("callfh: special call cannot use dynamic argument");
|
||||||
|
|
||||||
for(auto& i:ref_func.key_table)
|
for(auto& i:func.keys)
|
||||||
{
|
{
|
||||||
if(ref_hash.count(i.first))
|
if(hash.count(i.first))
|
||||||
ref_closure[i.second+1]=ref_hash[i.first];
|
closure[i.second+1]=hash[i.first];
|
||||||
else if(ref_func.local[i.second+1/*1 is reserved for 'me'*/].type==vm_none)
|
else if(func.local[i.second+1/*1 is reserved for 'me'*/].type==vm_none)
|
||||||
die("callfh: lack argument(s): \""+i.first+"\"");
|
die("callfh: lack argument(s): \""+i.first+"\"");
|
||||||
}
|
}
|
||||||
|
|
||||||
--stack_top;// pop hash
|
--top;// pop hash
|
||||||
ret.push(pc);
|
ret.push(pc);
|
||||||
pc=ref_func.entry-1;
|
pc=func.entry-1;
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_callb()
|
inline void nasal_vm::opr_callb()
|
||||||
{
|
{
|
||||||
(++stack_top)[0]=(*builtin_func[imm[pc]].func)(gc.local.back().vec()->elems,gc);
|
(++top)[0]=(*builtin_func[imm[pc]].func)(gc.local.back().vec()->elems,gc);
|
||||||
if(stack_top[0].type==vm_none)
|
if(top[0].type==vm_none)
|
||||||
die("native function error.");
|
die("native function error.");
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_slcbegin()
|
inline void nasal_vm::opr_slcbegin()
|
||||||
{
|
{
|
||||||
// | slice_vector | <-- stack_top[0]
|
// | slice_vector | <-- top[0]
|
||||||
// ----------------
|
// ----------------
|
||||||
// | resource_vec | <-- stack_top[-1]
|
// | resource_vec | <-- top[-1]
|
||||||
// ----------------
|
// ----------------
|
||||||
(++stack_top)[0]=gc.alloc(vm_vec);
|
(++top)[0]=gc.alloc(vm_vec);
|
||||||
if(stack_top[-1].type!=vm_vec)
|
if(top[-1].type!=vm_vec)
|
||||||
die("slcbegin: must slice a vector");
|
die("slcbegin: must slice a vector");
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_slcend()
|
inline void nasal_vm::opr_slcend()
|
||||||
{
|
{
|
||||||
stack_top[-1]=stack_top[0];
|
top[-1]=top[0];
|
||||||
--stack_top;
|
--top;
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_slc()
|
inline void nasal_vm::opr_slc()
|
||||||
{
|
{
|
||||||
nasal_ref val=(stack_top--)[0];
|
nasal_ref val=(top--)[0];
|
||||||
nasal_ref res=stack_top[-1].vec()->get_val(val.to_number());
|
nasal_ref res=top[-1].vec()->get_val(val.to_number());
|
||||||
if(res.type==vm_none)
|
if(res.type==vm_none)
|
||||||
die("slc: index out of range:"+std::to_string(val.to_number()));
|
die("slc: index out of range:"+std::to_string(val.to_number()));
|
||||||
stack_top[0].vec()->elems.push_back(res);
|
top[0].vec()->elems.push_back(res);
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_slc2()
|
inline void nasal_vm::opr_slc2()
|
||||||
{
|
{
|
||||||
nasal_ref val2=(stack_top--)[0];
|
nasal_ref val2=(top--)[0];
|
||||||
nasal_ref val1=(stack_top--)[0];
|
nasal_ref val1=(top--)[0];
|
||||||
std::vector<nasal_ref>& ref=stack_top[-1].vec()->elems;
|
std::vector<nasal_ref>& ref=top[-1].vec()->elems;
|
||||||
std::vector<nasal_ref>& aim=stack_top[0].vec()->elems;
|
std::vector<nasal_ref>& aim=top[0].vec()->elems;
|
||||||
|
|
||||||
uint8_t type1=val1.type,type2=val2.type;
|
uint8_t type1=val1.type,type2=val2.type;
|
||||||
int num1=val1.to_number();
|
int num1=val1.to_number();
|
||||||
int num2=val2.to_number();
|
int num2=val2.to_number();
|
||||||
int ref_size=ref.size();
|
size_t size=ref.size();
|
||||||
if(type1==vm_nil && type2==vm_nil)
|
if(type1==vm_nil && type2==vm_nil)
|
||||||
{
|
{
|
||||||
num1=0;
|
num1=0;
|
||||||
num2=ref_size-1;
|
num2=size-1;
|
||||||
}
|
}
|
||||||
else if(type1==vm_nil && type2!=vm_nil)
|
else if(type1==vm_nil && type2!=vm_nil)
|
||||||
num1=num2<0? -ref_size:0;
|
num1=num2<0? -size:0;
|
||||||
else if(type1!=vm_nil && type2==vm_nil)
|
else if(type1!=vm_nil && type2==vm_nil)
|
||||||
num2=num1<0? -1:ref_size-1;
|
num2=num1<0? -1:size-1;
|
||||||
|
|
||||||
if(num1>=num2)
|
if(num1>=num2)
|
||||||
die("slc2: begin index must be less than end index");
|
die("slc2: begin index must be less than end index");
|
||||||
else if(num1<-ref_size || num1>=ref_size)
|
else if(num1<-size || num1>=size)
|
||||||
die("slc2: begin index out of range: "+std::to_string(num1));
|
die("slc2: begin index out of range: "+std::to_string(num1));
|
||||||
else if(num2<-ref_size || num2>=ref_size)
|
else if(num2<-size || num2>=size)
|
||||||
die("slc2: end index out of range: "+std::to_string(num2));
|
die("slc2: end index out of range: "+std::to_string(num2));
|
||||||
else
|
else
|
||||||
for(int i=num1;i<=num2;++i)
|
for(int i=num1;i<=num2;++i)
|
||||||
aim.push_back(i>=0?ref[i]:ref[i+ref_size]);
|
aim.push_back(i>=0?ref[i]:ref[i+size]);
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_mcallg()
|
inline void nasal_vm::opr_mcallg()
|
||||||
{
|
{
|
||||||
mem_addr=gc.val_stack+imm[pc];
|
mem_addr=gc.stack+imm[pc];
|
||||||
(++stack_top)[0]=mem_addr[0];
|
(++top)[0]=mem_addr[0];
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_mcalll()
|
inline void nasal_vm::opr_mcalll()
|
||||||
{
|
{
|
||||||
mem_addr=&(gc.local.back().vec()->elems[imm[pc]]);
|
mem_addr=&(gc.local.back().vec()->elems[imm[pc]]);
|
||||||
(++stack_top)[0]=mem_addr[0];
|
(++top)[0]=mem_addr[0];
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_mupval()
|
inline void nasal_vm::opr_mupval()
|
||||||
{
|
{
|
||||||
mem_addr=&func_stk.top()->upvalue[(imm[pc]>>16)&0xffff].vec()->elems[imm[pc]&0xffff];
|
mem_addr=&func_stk.top()->upvalue[(imm[pc]>>16)&0xffff].vec()->elems[imm[pc]&0xffff];
|
||||||
(++stack_top)[0]=mem_addr[0];
|
(++top)[0]=mem_addr[0];
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_mcallv()
|
inline void nasal_vm::opr_mcallv()
|
||||||
{
|
{
|
||||||
nasal_ref val=stack_top[0];
|
nasal_ref val=top[0];
|
||||||
nasal_ref vec_addr=(--stack_top)[0];
|
nasal_ref vec=(--top)[0];
|
||||||
if(vec_addr.type==vm_vec)
|
if(vec.type==vm_vec)
|
||||||
{
|
{
|
||||||
mem_addr=vec_addr.vec()->get_mem(val.to_number());
|
mem_addr=vec.vec()->get_mem(val.to_number());
|
||||||
if(!mem_addr)
|
if(!mem_addr)
|
||||||
die("mcallv: index out of range:"+std::to_string(val.to_number()));
|
die("mcallv: index out of range:"+std::to_string(val.to_number()));
|
||||||
}
|
}
|
||||||
else if(vec_addr.type==vm_hash)
|
else if(vec.type==vm_hash)
|
||||||
{
|
{
|
||||||
if(val.type!=vm_str)
|
if(val.type!=vm_str)
|
||||||
die("mcallv: must use string as the key");
|
die("mcallv: must use string as the key");
|
||||||
nasal_hash& ref=*vec_addr.hash();
|
nasal_hash& ref=*vec.hash();
|
||||||
std::string& str=*val.str();
|
std::string& str=*val.str();
|
||||||
mem_addr=ref.get_mem(str);
|
mem_addr=ref.get_mem(str);
|
||||||
if(!mem_addr)
|
if(!mem_addr)
|
||||||
|
@ -798,10 +807,10 @@ inline void nasal_vm::opr_mcallv()
|
||||||
}
|
}
|
||||||
inline void nasal_vm::opr_mcallh()
|
inline void nasal_vm::opr_mcallh()
|
||||||
{
|
{
|
||||||
nasal_ref hash_addr=stack_top[0];
|
nasal_ref hash=top[0];
|
||||||
if(hash_addr.type!=vm_hash)
|
if(hash.type!=vm_hash)
|
||||||
die("mcallh: must call a hash");
|
die("mcallh: must call a hash");
|
||||||
nasal_hash& ref=*hash_addr.hash();
|
nasal_hash& ref=*hash.hash();
|
||||||
std::string& str=str_table[imm[pc]];
|
std::string& str=str_table[imm[pc]];
|
||||||
mem_addr=ref.get_mem(str);
|
mem_addr=ref.get_mem(str);
|
||||||
if(!mem_addr) // create a new key
|
if(!mem_addr) // create a new key
|
||||||
|
@ -814,9 +823,9 @@ inline void nasal_vm::opr_ret()
|
||||||
{
|
{
|
||||||
// get nasal_func and set 'me' to nil
|
// get nasal_func and set 'me' to nil
|
||||||
// and rewrite nasal_func with returned value
|
// and rewrite nasal_func with returned value
|
||||||
stack_top[-1].func()->local[0]={vm_nil};
|
top[-1].func()->local[0]={vm_nil};
|
||||||
--stack_top;
|
--top;
|
||||||
stack_top[0]=stack_top[1];
|
top[0]=top[1];
|
||||||
|
|
||||||
func_stk.pop(); // pop function stack
|
func_stk.pop(); // pop function stack
|
||||||
gc.local.pop_back(); // pop local scope
|
gc.local.pop_back(); // pop local scope
|
||||||
|
@ -826,7 +835,7 @@ void nasal_vm::run(
|
||||||
const nasal_codegen& gen,
|
const nasal_codegen& gen,
|
||||||
const nasal_import& linker,
|
const nasal_import& linker,
|
||||||
const bool opcnt,
|
const bool opcnt,
|
||||||
const bool debug=false)
|
const bool debug)
|
||||||
{
|
{
|
||||||
detail_info=debug;
|
detail_info=debug;
|
||||||
init(gen.get_strs(),gen.get_nums(),linker.get_file());
|
init(gen.get_strs(),gen.get_nums(),linker.get_file());
|
||||||
|
@ -863,7 +872,7 @@ void nasal_vm::run(
|
||||||
}
|
}
|
||||||
|
|
||||||
// set canary and program counter
|
// set canary and program counter
|
||||||
auto& canary=gc.val_stack[STACK_MAX_DEPTH-1];
|
auto& canary=gc.stack[STACK_MAX_DEPTH-1];
|
||||||
canary.value.gcobj=nullptr;
|
canary.value.gcobj=nullptr;
|
||||||
pc=0;
|
pc=0;
|
||||||
// goto the first operand
|
// goto the first operand
|
||||||
|
|
Loading…
Reference in New Issue