This commit is contained in:
Valk Richard Li 2020-12-19 21:02:02 +08:00
parent 9f30440286
commit fed1e20085
3 changed files with 83 additions and 82 deletions

View File

@ -95,7 +95,6 @@ public:
~nasal_bytecode_vm();
void clear();
void run(std::vector<std::string>&,std::vector<double>&,std::vector<opcode>&);
void nas_switch_threading(std::vector<std::string>&,std::vector<double>&,std::vector<opcode>&);
};
nasal_bytecode_vm::nasal_bytecode_vm()
@ -729,17 +728,24 @@ void nasal_bytecode_vm::opr_call()
{
int val_addr=-1;
int name_index=exec_code[ptr].index;
if(local_scope_stack.top()>=0)
val_addr=vm.gc_get(local_scope_stack.top()).get_closure().get_value_address(name_index);
if(val_addr<0)
val_addr=vm.gc_get(global_scope_addr).get_closure().get_value_address(name_index);
if(val_addr<0)
int local_scope_top=local_scope_stack.top();
if(local_scope_top>=0)
val_addr=vm.gc_get(local_scope_top).get_closure().get_value_address(name_index);
if(val_addr>=0)
{
die("call: cannot find symbol named \""+string_table[name_index]+"\"");
vm.add_reference(val_addr);
*(++value_stack_top)=val_addr;
return;
}
vm.add_reference(val_addr);
*(++value_stack_top)=val_addr;
else
val_addr=vm.gc_get(global_scope_addr).get_closure().get_value_address(name_index);
if(val_addr>=0)
{
vm.add_reference(val_addr);
*(++value_stack_top)=val_addr;
return;
}
die("call: cannot find symbol named \""+string_table[name_index]+"\"");
return;
}
void nasal_bytecode_vm::opr_callv()
@ -751,27 +757,16 @@ void nasal_bytecode_vm::opr_callv()
{
int num=vm.gc_get(val_addr).to_number();
int res=vm.gc_get(vec_addr).get_vector().get_value_address(num);
if(res<0)
if(res>=0)
{
vm.add_reference(res);
*(++value_stack_top)=res;
}
else
{
die("callv: index out of range");
return;
}
vm.add_reference(res);
*(++value_stack_top)=res;
}
else if(type==vm_string)
{
std::string str=vm.gc_get(vec_addr).get_string();
int num=vm.gc_get(val_addr).to_number();
int str_size=str.length();
if(num<-str_size || num>=str_size)
{
die("callv: index out of range");
return;
}
int res=vm.gc_alloc(vm_number);
vm.gc_get(res).set_number((double)str[(num+str_size)%str_size]);
*(++value_stack_top)=res;
}
else if(type==vm_hash)
{
@ -794,6 +789,20 @@ void nasal_bytecode_vm::opr_callv()
vm.add_reference(res);
*(++value_stack_top)=res;
}
else if(type==vm_string)
{
std::string str=vm.gc_get(vec_addr).get_string();
int num=vm.gc_get(val_addr).to_number();
int str_size=str.length();
if(num<-str_size || num>=str_size)
{
die("callv: index out of range");
return;
}
int res=vm.gc_alloc(vm_number);
vm.gc_get(res).set_number((double)str[num>=0? num:num+str_size]);
*(++value_stack_top)=res;
}
vm.del_reference(val_addr);
vm.del_reference(vec_addr);
return;
@ -825,12 +834,13 @@ void nasal_bytecode_vm::opr_callh()
return;
}
int res=vm.gc_get(val_addr).get_hash().get_value_address(string_table[exec_code[ptr].index]);
if(res<0)
if(res>=0)
vm.add_reference(res);
else
{
die("callh: hash member \""+string_table[exec_code[ptr].index]+"\" does not exist");
return;
}
vm.add_reference(res);
if(vm.gc_get(res).get_type()==vm_function)
vm.gc_get(vm.gc_get(res).get_func().get_closure_addr()).get_closure().add_new_value(me_index,val_addr);
else
@ -931,8 +941,9 @@ void nasal_bytecode_vm::opr_builtincall()
void nasal_bytecode_vm::opr_slicebegin()
{
slice_stack.push(vm.gc_alloc(vm_vector));
if(vm.gc_get(*value_stack_top).get_type()!=vm_vector)
die("slcbegin: must slice a vector");
if(vm.gc_get(*value_stack_top).get_type()==vm_vector)
return;
die("slcbegin: must slice a vector");
return;
}
void nasal_bytecode_vm::opr_sliceend()
@ -954,14 +965,14 @@ void nasal_bytecode_vm::opr_slice()
default:die("slc: error value type");break;
}
int res=vm.gc_get(*value_stack_top).get_vector().get_value_address((int)num);
if(res<0)
if(res>=0)
{
die("slc: index out of range");
vm.add_reference(res);
vm.gc_get(slice_stack.top()).get_vector().add_elem(res);
vm.del_reference(val_addr);
return;
}
vm.add_reference(res);
vm.gc_get(slice_stack.top()).get_vector().add_elem(res);
vm.del_reference(val_addr);
die("slc: index out of range");
return;
}
void nasal_bytecode_vm::opr_slice2()
@ -1023,13 +1034,22 @@ void nasal_bytecode_vm::opr_mcall()
{
int* mem_addr=NULL;
int name_index=exec_code[ptr].index;
if(local_scope_stack.top()>=0)
mem_addr=vm.gc_get(local_scope_stack.top()).get_closure().get_mem_address(name_index);
if(!mem_addr)
int local_scope_top=local_scope_stack.top();
if(local_scope_top>=0)
mem_addr=vm.gc_get(local_scope_top).get_closure().get_mem_address(name_index);
if(mem_addr)
{
pointer_stack.push(mem_addr);
return;
}
else
mem_addr=vm.gc_get(global_scope_addr).get_closure().get_mem_address(name_index);
if(!mem_addr)
die("mcall: cannot find symbol named \""+string_table[name_index]+"\"");
pointer_stack.push(mem_addr);
if(mem_addr)
{
pointer_stack.push(mem_addr);
return;
}
die("mcall: cannot find symbol named \""+string_table[name_index]+"\"");
return;
}
void nasal_bytecode_vm::opr_mcallv()
@ -1038,11 +1058,6 @@ void nasal_bytecode_vm::opr_mcallv()
int* vec_addr=pointer_stack.top();
pointer_stack.pop();
int type=vm.gc_get(*vec_addr).get_type();
if(type==vm_string)
{
die("mcallv: cannot get memory space in a string");
return;
}
if(type==vm_vector)
{
int num;
@ -1077,6 +1092,11 @@ void nasal_bytecode_vm::opr_mcallv()
}
pointer_stack.push(res);
}
else
{
die("mcallv: cannot get memory space in a string");
return;
}
vm.del_reference(val_addr);
return;
}

View File

@ -114,8 +114,8 @@ protected:
nasal_function* func;
nasal_closure* cls;
}ptr;
public:
int ref_cnt;
nasal_scalar();
~nasal_scalar();
void clear();
@ -135,20 +135,10 @@ public:
class nasal_virtual_machine
{
struct gc_unit
{
int ref_cnt;
nasal_scalar elem;
gc_unit()
{
ref_cnt=0;
return;
}
};
private:
nasal_scalar error_returned_value;
std::queue<int> garbage_collector_free_space;
std::vector<gc_unit*> garbage_collector_memory;
std::vector<nasal_scalar*> garbage_collector_memory;
public:
~nasal_virtual_machine();
void clear();
@ -646,10 +636,7 @@ nasal_virtual_machine::~nasal_virtual_machine()
int gc_mem_size=garbage_collector_memory.size();
for(int i=0;i<gc_mem_size;++i)
if(garbage_collector_memory[i]->ref_cnt)
{
garbage_collector_memory[i]->ref_cnt=0;
garbage_collector_memory[i]->elem.clear();
}
garbage_collector_memory[i]->clear();
for(int i=0;i<gc_mem_size;++i)
delete garbage_collector_memory[i];
while(!garbage_collector_free_space.empty())
@ -664,11 +651,11 @@ void nasal_virtual_machine::debug()
if(garbage_collector_memory[i]->ref_cnt)
{
std::cout<<">> [debug] "<<i<<": "<<garbage_collector_memory[i]->ref_cnt<<" ";
switch(garbage_collector_memory[i]->elem.get_type())
switch(garbage_collector_memory[i]->get_type())
{
case vm_nil:std::cout<<"nil";break;
case vm_number:std::cout<<"number "<<garbage_collector_memory[i]->elem.get_number();break;
case vm_string:std::cout<<"string "<<garbage_collector_memory[i]->elem.get_string();break;
case vm_number:std::cout<<"number "<<garbage_collector_memory[i]->get_number();break;
case vm_string:std::cout<<"string "<<garbage_collector_memory[i]->get_string();break;
case vm_vector:std::cout<<"vector";break;
case vm_hash:std::cout<<"hash";break;
case vm_function:std::cout<<"function";break;
@ -683,10 +670,7 @@ void nasal_virtual_machine::clear()
int gc_mem_size=garbage_collector_memory.size();
for(int i=0;i<gc_mem_size;++i)
if(garbage_collector_memory[i]->ref_cnt)
{
garbage_collector_memory[i]->ref_cnt=0;
garbage_collector_memory[i]->elem.clear();
}
garbage_collector_memory[i]->clear();
for(int i=0;i<gc_mem_size;++i)
delete garbage_collector_memory[i];
while(!garbage_collector_free_space.empty())
@ -699,41 +683,41 @@ int nasal_virtual_machine::gc_alloc(int val_type)
if(garbage_collector_free_space.empty())
{
int mem_size=garbage_collector_memory.size();
gc_unit* new_unit=new gc_unit;
nasal_scalar* new_unit=new nasal_scalar;
garbage_collector_memory.push_back(new_unit);
int ret=mem_size;
new_unit->ref_cnt=1;
new_unit->elem.set_type(val_type,*this);
new_unit->set_type(val_type,*this);
return ret;
}
int ret=garbage_collector_free_space.front();
gc_unit& unit_ref=*garbage_collector_memory[ret];
nasal_scalar& unit_ref=*garbage_collector_memory[ret];
unit_ref.ref_cnt=1;
unit_ref.elem.set_type(val_type,*this);
unit_ref.set_type(val_type,*this);
garbage_collector_free_space.pop();
return ret;
}
nasal_scalar& nasal_virtual_machine::gc_get(int value_address)
{
if(0<=value_address)
return garbage_collector_memory[value_address]->elem;
if(value_address>=0)
return *garbage_collector_memory[value_address];
return error_returned_value;
}
void nasal_virtual_machine::add_reference(int value_address)
{
if(0<=value_address)
if(value_address>=0)
++garbage_collector_memory[value_address]->ref_cnt;
return;
}
void nasal_virtual_machine::del_reference(int value_address)
{
if(0<=value_address)
if(value_address>=0)
--garbage_collector_memory[value_address]->ref_cnt;
else
return;
if(!garbage_collector_memory[value_address]->ref_cnt)
{
garbage_collector_memory[value_address]->elem.clear();
garbage_collector_memory[value_address]->clear();
garbage_collector_free_space.push(value_address);
}
return;

View File

@ -1061,10 +1061,7 @@ nasal_ast nasal_parse::unary()
if(node.get_children()[0].get_type()==ast_number)
{
double num=node.get_children()[0].get_num();
if(node.get_type()==ast_unary_not)
num=(!num);
else
num=-num;
num=(node.get_type()==ast_unary_not?(!num):-num);
node.set_type(ast_number);
node.set_num(num);
node.get_children().clear();