2020-10-23 14:53:04 +08:00
|
|
|
#ifndef __NASAL_BUILTIN_H__
|
|
|
|
#define __NASAL_BUILTIN_H__
|
2021-03-27 01:08:05 +08:00
|
|
|
/*
|
|
|
|
builtin functions must be called inside a outer function like this:
|
|
|
|
var print=func(elements...)
|
|
|
|
{
|
|
|
|
__builtin_std_cout(elements);
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
builtin function __builtin_std_cout is wrapped up by print
|
|
|
|
*/
|
2020-10-23 14:53:04 +08:00
|
|
|
|
2020-11-20 19:15:12 +08:00
|
|
|
// declaration of builtin functions
|
|
|
|
// to add new builtin function,declare it here and write the definition below
|
2020-12-24 23:53:31 +08:00
|
|
|
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_print(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_append(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_setsize(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_system(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_input(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_sleep(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_fin(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_fout(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_split(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_rand(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_id(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_int(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_num(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_pop(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_str(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_size(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_xor(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_and(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_or(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_nand(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_not(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_sin(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_cos(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_tan(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_exp(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_ln(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_sqrt(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_atan2(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_time(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_contains(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_delete(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_getkeys(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_import(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_die(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_type(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_substr(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_streq(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_left(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_right(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_cmp(std::vector<nasal_val*>&,nasal_gc&);
|
|
|
|
nasal_val* builtin_chr(std::vector<nasal_val*>&,nasal_gc&);
|
2020-11-20 19:15:12 +08:00
|
|
|
|
2021-03-28 17:39:24 +08:00
|
|
|
void builtin_err(std::string func_name,std::string info)
|
2020-12-24 23:53:31 +08:00
|
|
|
{
|
|
|
|
std::cout<<">> [vm] "<<func_name<<": "<<info<<".\n";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-11-20 19:15:12 +08:00
|
|
|
// register builtin function's name and it's address here in this table below
|
2021-02-12 22:27:41 +08:00
|
|
|
// this table must end with {"",NULL}
|
2020-11-20 19:15:12 +08:00
|
|
|
struct FUNC_TABLE
|
|
|
|
{
|
2021-03-27 01:08:05 +08:00
|
|
|
std::string name;
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* (*func)(std::vector<nasal_val*>&,nasal_gc&);
|
2021-03-27 01:08:05 +08:00
|
|
|
} builtin_func[]=
|
|
|
|
{
|
|
|
|
{"__builtin_std_cout", builtin_print },
|
|
|
|
{"__builtin_push_back",builtin_append },
|
|
|
|
{"__builtin_set_size", builtin_setsize },
|
|
|
|
{"__builtin_system", builtin_system },
|
|
|
|
{"__builtin_input", builtin_input },
|
|
|
|
{"__builtin_sleep", builtin_sleep },
|
|
|
|
{"__builtin_fin", builtin_fin },
|
|
|
|
{"__builtin_fout", builtin_fout },
|
|
|
|
{"__builtin_split", builtin_split },
|
|
|
|
{"__builtin_rand", builtin_rand },
|
|
|
|
{"__builtin_get_id", builtin_id },
|
|
|
|
{"__builtin_int", builtin_int },
|
|
|
|
{"__builtin_num", builtin_num },
|
|
|
|
{"__builtin_pop_back", builtin_pop },
|
|
|
|
{"__builtin_str", builtin_str },
|
|
|
|
{"__builtin_size", builtin_size },
|
|
|
|
{"__builtin_xor", builtin_xor },
|
|
|
|
{"__builtin_and", builtin_and },
|
|
|
|
{"__builtin_or", builtin_or },
|
|
|
|
{"__builtin_nand", builtin_nand },
|
|
|
|
{"__builtin_not", builtin_not },
|
|
|
|
{"__builtin_sin", builtin_sin },
|
|
|
|
{"__builtin_cos", builtin_cos },
|
|
|
|
{"__builtin_tan", builtin_tan },
|
|
|
|
{"__builtin_exp", builtin_exp },
|
|
|
|
{"__builtin_ln", builtin_ln },
|
|
|
|
{"__builtin_sqrt", builtin_sqrt },
|
|
|
|
{"__builtin_atan2", builtin_atan2 },
|
|
|
|
{"__builtin_time", builtin_time },
|
|
|
|
{"__builtin_contains", builtin_contains},
|
|
|
|
{"__builtin_delete", builtin_delete },
|
|
|
|
{"__builtin_get_keys", builtin_getkeys },
|
|
|
|
{"__builtin_import", builtin_import },
|
|
|
|
{"__builtin_die", builtin_die },
|
|
|
|
{"__builtin_type", builtin_type },
|
|
|
|
{"__builtin_substr", builtin_substr },
|
|
|
|
{"__builtin_streq", builtin_streq },
|
|
|
|
{"__builtin_left", builtin_left },
|
|
|
|
{"__builtin_right", builtin_right },
|
|
|
|
{"__builtin_cmp", builtin_cmp },
|
|
|
|
{"__builtin_chr", builtin_chr },
|
|
|
|
{"", nullptr }
|
2020-11-20 19:15:12 +08:00
|
|
|
};
|
|
|
|
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_print(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
|
|
|
// get arguments
|
2021-04-19 19:12:41 +08:00
|
|
|
// local_scope[0] is reserved for 'me'
|
|
|
|
nasal_val* vec_addr=local_scope[1];
|
2020-10-23 14:53:04 +08:00
|
|
|
// main process
|
2021-05-04 17:39:24 +08:00
|
|
|
for(auto i:vec_addr->ptr.vec->elems)
|
|
|
|
switch(i->type)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-27 01:08:05 +08:00
|
|
|
case vm_nil: std::cout<<"nil"; break;
|
2021-05-04 17:39:24 +08:00
|
|
|
case vm_num: std::cout<<i->ptr.num; break;
|
|
|
|
case vm_str: std::cout<<*i->ptr.str; break;
|
|
|
|
case vm_vec: i->ptr.vec->print(); break;
|
|
|
|
case vm_hash: i->ptr.hash->print(); break;
|
2021-03-27 01:08:05 +08:00
|
|
|
case vm_func: std::cout<<"func(...){...}"; break;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
|
|
|
// generate return value
|
2021-03-30 00:12:48 +08:00
|
|
|
return gc.nil_addr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_append(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* vec_addr=local_scope[1];
|
|
|
|
nasal_val* elem_addr=local_scope[2];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(vec_addr->type!=vm_vec)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("append","\"vector\" must be vector");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
std::vector<nasal_val*>& ref_vec=vec_addr->ptr.vec->elems;
|
2021-05-04 17:39:24 +08:00
|
|
|
for(auto i:elem_addr->ptr.vec->elems)
|
|
|
|
ref_vec.push_back(i);
|
2021-03-30 00:12:48 +08:00
|
|
|
return gc.nil_addr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_setsize(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* vec_addr=local_scope[1];
|
|
|
|
nasal_val* size_addr=local_scope[2];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(vec_addr->type!=vm_vec)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("setsize","\"vector\" must be vector");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
if(size_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("setsize","\"size\" is not a number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
int num=size_addr->ptr.num;
|
|
|
|
if(num<0)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("setsize","\"size\" must be greater than -1");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-05-04 17:39:24 +08:00
|
|
|
vec_addr->ptr.vec->elems.resize(num,gc.nil_addr);
|
2021-03-30 00:12:48 +08:00
|
|
|
return gc.nil_addr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
|
|
|
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_system(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-06-03 21:49:31 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* str_addr=local_scope[1];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(str_addr->type!=vm_str)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("system","\"str\" must be string");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-06-03 21:49:31 +08:00
|
|
|
ret_addr->ptr.num=(double)system(str_addr->ptr.str->data());
|
|
|
|
return ret_addr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
|
|
|
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_input(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_str);
|
2021-03-27 01:08:05 +08:00
|
|
|
std::cin>>*ret_addr->ptr.str;
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
|
|
|
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_sleep(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("sleep","\"duration\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-02-10 00:12:22 +08:00
|
|
|
// sleep in unistd.h will make this progress sleep sleep_time seconds.
|
2021-03-28 17:39:24 +08:00
|
|
|
sleep((unsigned long)val_addr->ptr.num);
|
2021-03-30 00:12:48 +08:00
|
|
|
return gc.nil_addr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
|
|
|
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_fin(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type!=vm_str)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("io.fin","\"filename\" must be string");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
std::string& filename=*val_addr->ptr.str;
|
2020-10-23 14:53:04 +08:00
|
|
|
std::ifstream fin(filename);
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_str);
|
2021-03-27 01:08:05 +08:00
|
|
|
*ret_addr->ptr.str="";
|
2020-10-23 14:53:04 +08:00
|
|
|
if(!fin.fail())
|
|
|
|
while(!fin.eof())
|
|
|
|
{
|
2020-11-03 19:27:21 +08:00
|
|
|
char c=fin.get();
|
2020-10-23 14:53:04 +08:00
|
|
|
if(fin.eof())
|
|
|
|
break;
|
2021-03-27 01:08:05 +08:00
|
|
|
ret_addr->ptr.str->push_back(c);
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
|
|
|
else
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("io.fin","cannot open \""+filename+"\"");
|
2020-10-23 14:53:04 +08:00
|
|
|
fin.close();
|
|
|
|
return ret_addr;
|
|
|
|
}
|
|
|
|
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_fout(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
|
|
|
nasal_val* str_addr=local_scope[2];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type!=vm_str)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("io.fout","\"filename\" must be string");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
if(str_addr->type!=vm_str)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("io.fout","\"str\" must be string");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
std::ofstream fout(*val_addr->ptr.str);
|
2021-03-27 01:08:05 +08:00
|
|
|
if(fout.fail())
|
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("io.fout","cannot open \""+*val_addr->ptr.str+"\"");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
fout<<*str_addr->ptr.str;
|
2020-10-23 14:53:04 +08:00
|
|
|
fout.close();
|
2021-03-30 00:12:48 +08:00
|
|
|
return gc.nil_addr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
|
|
|
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_split(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* delimeter_val_addr=local_scope[1];
|
|
|
|
nasal_val* string_val_addr=local_scope[2];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(delimeter_val_addr->type!=vm_str)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("split","\"delimeter\" must be string");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
if(string_val_addr->type!=vm_str)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("split","\"string\" must be string");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
std::string delimeter=*delimeter_val_addr->ptr.str;
|
|
|
|
std::string source=*string_val_addr->ptr.str;
|
2020-10-23 14:53:04 +08:00
|
|
|
int delimeter_len=delimeter.length();
|
|
|
|
int source_len=source.length();
|
2021-05-31 19:10:59 +08:00
|
|
|
nasal_val* ret_addr=gc.builtin_alloc(vm_vec);
|
2021-03-28 17:39:24 +08:00
|
|
|
std::vector<nasal_val*>& ref_vec=ret_addr->ptr.vec->elems;
|
2020-10-23 14:53:04 +08:00
|
|
|
std::string tmp="";
|
2020-10-25 22:15:49 +08:00
|
|
|
|
|
|
|
if(!delimeter_len)
|
|
|
|
{
|
|
|
|
for(int i=0;i<source_len;++i)
|
|
|
|
{
|
2021-05-31 19:10:59 +08:00
|
|
|
nasal_val* str_addr=gc.builtin_alloc(vm_str);
|
2021-03-27 01:08:05 +08:00
|
|
|
*str_addr->ptr.str=source[i];
|
2021-03-28 17:39:24 +08:00
|
|
|
ref_vec.push_back(str_addr);
|
2020-10-25 22:15:49 +08:00
|
|
|
}
|
|
|
|
return ret_addr;
|
|
|
|
}
|
|
|
|
|
2020-10-23 14:53:04 +08:00
|
|
|
for(int i=0;i<source_len;++i)
|
|
|
|
{
|
|
|
|
bool check_delimeter=false;
|
|
|
|
if(source[i]==delimeter[0])
|
|
|
|
for(int j=0;j<delimeter_len;++j)
|
|
|
|
{
|
|
|
|
if(i+j>=source_len || source[i+j]!=delimeter[j])
|
|
|
|
break;
|
|
|
|
if(j==delimeter_len-1)
|
|
|
|
check_delimeter=true;
|
|
|
|
}
|
|
|
|
if(check_delimeter)
|
|
|
|
{
|
2021-02-12 22:27:41 +08:00
|
|
|
if(tmp.length())
|
|
|
|
{
|
2021-05-31 19:10:59 +08:00
|
|
|
nasal_val* str_addr=gc.builtin_alloc(vm_str);
|
2021-03-27 01:08:05 +08:00
|
|
|
*str_addr->ptr.str=tmp;
|
2021-03-28 17:39:24 +08:00
|
|
|
ref_vec.push_back(str_addr);
|
2021-02-12 22:27:41 +08:00
|
|
|
tmp="";
|
|
|
|
}
|
2020-10-23 14:53:04 +08:00
|
|
|
i+=delimeter_len-1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
tmp+=source[i];
|
|
|
|
}
|
|
|
|
if(tmp.length())
|
|
|
|
{
|
2021-05-31 19:10:59 +08:00
|
|
|
nasal_val* str_addr=gc.builtin_alloc(vm_str);
|
2021-03-27 01:08:05 +08:00
|
|
|
*str_addr->ptr.str=tmp;
|
2021-03-28 17:39:24 +08:00
|
|
|
ref_vec.push_back(str_addr);
|
2020-10-23 14:53:04 +08:00
|
|
|
tmp="";
|
|
|
|
}
|
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_rand(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type!=vm_num && val_addr->type!=vm_nil)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("rand","\"seed\" must be nil or number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type==vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
srand((unsigned int)val_addr->ptr.num);
|
2021-03-30 00:12:48 +08:00
|
|
|
return gc.nil_addr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
|
|
|
double num=0;
|
|
|
|
for(int i=0;i<5;++i)
|
|
|
|
num=(num+rand())*(1.0/(RAND_MAX+1.0));
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-27 01:08:05 +08:00
|
|
|
ret_addr->ptr.num=num;
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_id(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_str);
|
2020-12-19 23:47:04 +08:00
|
|
|
char buf[32];
|
2021-03-28 17:39:24 +08:00
|
|
|
sprintf(buf,"0x%p",val_addr);
|
2021-03-27 01:08:05 +08:00
|
|
|
*ret_addr->ptr.str=buf;
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_int(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("int","\"value\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
int number=(int)val_addr->ptr.num;
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-27 01:08:05 +08:00
|
|
|
ret_addr->ptr.num=(double)number;
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_num(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type!=vm_str)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("num","\"value\" must be string");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-28 17:39:24 +08:00
|
|
|
ret_addr->ptr.num=val_addr->to_number();
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_pop(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type!=vm_vec)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("pop","\"vector\" must be vector");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->ptr.vec->elems.size())
|
2021-03-27 01:08:05 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
nasal_val* tmp=val_addr->ptr.vec->elems.back();
|
|
|
|
val_addr->ptr.vec->elems.pop_back();
|
2021-03-27 01:08:05 +08:00
|
|
|
return tmp;
|
|
|
|
}
|
2021-03-30 00:12:48 +08:00
|
|
|
return gc.nil_addr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_str(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("str","\"number\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_str);
|
2021-03-28 17:39:24 +08:00
|
|
|
*ret_addr->ptr.str=val_addr->to_string();
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_size(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-28 17:39:24 +08:00
|
|
|
switch(val_addr->type)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
case vm_nil: ret_addr->ptr.num=0; break;
|
|
|
|
case vm_num: ret_addr->ptr.num=val_addr->ptr.num; break;
|
|
|
|
case vm_func: ret_addr->ptr.num=0; break;
|
|
|
|
case vm_str: ret_addr->ptr.num=val_addr->ptr.str->length(); break;
|
|
|
|
case vm_vec: ret_addr->ptr.num=val_addr->ptr.vec->elems.size(); break;
|
|
|
|
case vm_hash: ret_addr->ptr.num=val_addr->ptr.hash->elems.size();break;
|
2021-02-10 00:12:22 +08:00
|
|
|
}
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_xor(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* a_addr=local_scope[1];
|
|
|
|
nasal_val* b_addr=local_scope[2];
|
2021-03-27 01:08:05 +08:00
|
|
|
if(a_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("xor","\"a\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-27 01:08:05 +08:00
|
|
|
if(b_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("xor","\"b\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-27 01:08:05 +08:00
|
|
|
int number_a=(int)a_addr->ptr.num;
|
|
|
|
int number_b=(int)b_addr->ptr.num;
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-27 01:08:05 +08:00
|
|
|
ret_addr->ptr.num=(number_a^number_b);
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_and(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* a_addr=local_scope[1];
|
|
|
|
nasal_val* b_addr=local_scope[2];
|
2021-03-27 01:08:05 +08:00
|
|
|
if(a_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("and","\"a\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-27 01:08:05 +08:00
|
|
|
if(b_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("and","\"b\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-27 01:08:05 +08:00
|
|
|
int number_a=(int)a_addr->ptr.num;
|
|
|
|
int number_b=(int)b_addr->ptr.num;
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-27 01:08:05 +08:00
|
|
|
ret_addr->ptr.num=(number_a&number_b);
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_or(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* a_addr=local_scope[1];
|
|
|
|
nasal_val* b_addr=local_scope[2];
|
2021-03-27 01:08:05 +08:00
|
|
|
if(a_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("or","\"a\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-27 01:08:05 +08:00
|
|
|
if(b_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("or","\"b\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-27 01:08:05 +08:00
|
|
|
int number_a=(int)a_addr->ptr.num;
|
|
|
|
int number_b=(int)b_addr->ptr.num;
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-27 01:08:05 +08:00
|
|
|
ret_addr->ptr.num=(number_a|number_b);
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_nand(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* a_addr=local_scope[1];
|
|
|
|
nasal_val* b_addr=local_scope[2];
|
2021-03-27 01:08:05 +08:00
|
|
|
if(a_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("nand","\"a\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-27 01:08:05 +08:00
|
|
|
if(b_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("nand","\"b\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-27 01:08:05 +08:00
|
|
|
int number_a=(int)a_addr->ptr.num;
|
|
|
|
int number_b=(int)b_addr->ptr.num;
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-27 01:08:05 +08:00
|
|
|
ret_addr->ptr.num=(~(number_a&number_b));
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_not(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* a_addr=local_scope[1];
|
2021-03-27 01:08:05 +08:00
|
|
|
if(a_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("not","\"a\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-27 01:08:05 +08:00
|
|
|
int number=(int)a_addr->ptr.num;
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-27 01:08:05 +08:00
|
|
|
ret_addr->ptr.num=(~number);
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_sin(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("sin","\"x\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-28 17:39:24 +08:00
|
|
|
ret_addr->ptr.num=sin(val_addr->ptr.num);
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_cos(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("cos","\"x\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-28 17:39:24 +08:00
|
|
|
ret_addr->ptr.num=cos(val_addr->ptr.num);
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_tan(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("tan","\"x\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-28 17:39:24 +08:00
|
|
|
ret_addr->ptr.num=tan(val_addr->ptr.num);
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_exp(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("exp","\"x\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-28 17:39:24 +08:00
|
|
|
ret_addr->ptr.num=exp(val_addr->ptr.num);
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_ln(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("ln","\"x\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-28 17:39:24 +08:00
|
|
|
ret_addr->ptr.num=(log(val_addr->ptr.num)/log(2.7182818284590452354));
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_sqrt(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("sqrt","\"x\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-28 17:39:24 +08:00
|
|
|
ret_addr->ptr.num=sqrt(val_addr->ptr.num);
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_atan2(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* x_val_addr=local_scope[1];
|
2021-05-31 19:10:59 +08:00
|
|
|
nasal_val* y_val_addr=local_scope[2];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(x_val_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("atan2","\"x\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
if(y_val_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("atan2","\"y\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-28 17:39:24 +08:00
|
|
|
ret_addr->ptr.num=atan2(y_val_addr->ptr.num,x_val_addr->ptr.num);
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_time(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(val_addr->type!=vm_num)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("time","\"begin_time\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
time_t begin_time=(time_t)val_addr->ptr.num;
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-27 01:08:05 +08:00
|
|
|
ret_addr->ptr.num=time(&begin_time);
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_contains(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* hash_addr=local_scope[1];
|
|
|
|
nasal_val* key_addr=local_scope[2];
|
2021-03-27 01:08:05 +08:00
|
|
|
if(hash_addr->type!=vm_hash)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("contains","\"hash\" must be hash");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-27 01:08:05 +08:00
|
|
|
if(key_addr->type!=vm_str)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("contains","\"key\" must be string");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-05-31 19:10:59 +08:00
|
|
|
ret_addr->ptr.num=hash_addr->ptr.hash->elems.count(*key_addr->ptr.str);
|
2020-10-23 14:53:04 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_delete(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* hash_addr=local_scope[1];
|
|
|
|
nasal_val* key_addr=local_scope[2];
|
2021-03-27 01:08:05 +08:00
|
|
|
if(hash_addr->type!=vm_hash)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("delete","\"hash\" must be hash");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-27 01:08:05 +08:00
|
|
|
if(key_addr->type!=vm_str)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("delete","\"key\" must be string");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-03-27 01:08:05 +08:00
|
|
|
if(hash_addr->ptr.hash->elems.count(*key_addr->ptr.str))
|
|
|
|
hash_addr->ptr.hash->elems.erase(*key_addr->ptr.str);
|
2021-03-30 00:12:48 +08:00
|
|
|
return gc.nil_addr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_getkeys(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* hash_addr=local_scope[1];
|
2021-03-27 01:08:05 +08:00
|
|
|
if(hash_addr->type!=vm_hash)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("keys","\"hash\" must be hash");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-05-31 19:10:59 +08:00
|
|
|
nasal_val* ret_addr=gc.builtin_alloc(vm_vec);
|
2021-03-28 17:39:24 +08:00
|
|
|
std::vector<nasal_val*>& ref_vec=ret_addr->ptr.vec->elems;
|
2021-05-04 17:39:24 +08:00
|
|
|
for(auto iter:hash_addr->ptr.hash->elems)
|
2021-03-28 17:39:24 +08:00
|
|
|
{
|
2021-05-31 19:10:59 +08:00
|
|
|
nasal_val* str_addr=gc.builtin_alloc(vm_str);
|
2021-05-04 17:39:24 +08:00
|
|
|
*str_addr->ptr.str=iter.first;
|
2021-03-28 17:39:24 +08:00
|
|
|
ref_vec.push_back(str_addr);
|
|
|
|
}
|
|
|
|
return ret_addr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_import(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
|
|
|
// this function is used in preprocessing.
|
|
|
|
// this function will return nothing when running.
|
2021-05-04 17:39:24 +08:00
|
|
|
builtin_err("import","must use this function in global scope");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_die(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* str_addr=local_scope[1];
|
2021-03-27 01:08:05 +08:00
|
|
|
if(str_addr->type!=vm_str)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("die","\"str\" must be string");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-04-02 22:19:29 +08:00
|
|
|
std::cout<<">> [vm] error: "<<*str_addr->ptr.str<<'\n';
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_type(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* val_addr=local_scope[1];
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_str);
|
2021-03-28 17:39:24 +08:00
|
|
|
switch(val_addr->type)
|
2020-10-23 14:53:04 +08:00
|
|
|
{
|
2021-03-27 01:08:05 +08:00
|
|
|
case vm_nil: *ret_addr->ptr.str="nil"; break;
|
|
|
|
case vm_num: *ret_addr->ptr.str="number"; break;
|
|
|
|
case vm_str: *ret_addr->ptr.str="string"; break;
|
|
|
|
case vm_vec: *ret_addr->ptr.str="vector"; break;
|
|
|
|
case vm_hash: *ret_addr->ptr.str="hash"; break;
|
|
|
|
case vm_func: *ret_addr->ptr.str="function"; break;
|
2020-10-23 14:53:04 +08:00
|
|
|
}
|
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_substr(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2020-10-25 22:15:49 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* str_addr=local_scope[1];
|
|
|
|
nasal_val* beg_addr=local_scope[2];
|
|
|
|
nasal_val* len_addr=local_scope[3];
|
2021-03-27 01:08:05 +08:00
|
|
|
if(str_addr->type!=vm_str)
|
2020-10-25 22:15:49 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("substr","\"str\" must be string");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-25 22:15:49 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
if(beg_addr->type!=vm_num)
|
2020-10-25 22:15:49 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("substr","\"begin\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-25 22:15:49 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
if(len_addr->type!=vm_num)
|
2020-10-25 22:15:49 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("substr","\"length\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-25 22:15:49 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
std::string& str=*str_addr->ptr.str;
|
|
|
|
int beg=(int)beg_addr->ptr.num;
|
|
|
|
int len=(int)len_addr->ptr.num;
|
|
|
|
if(beg>=str.length() || beg+len>=str.length())
|
2020-10-25 22:15:49 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("susbtr","index out of range");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2020-10-25 22:15:49 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
if(len<0)
|
|
|
|
len=0;
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_str);
|
2021-03-28 17:39:24 +08:00
|
|
|
*ret_addr->ptr.str=str.substr(beg,len);
|
2020-10-25 22:15:49 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_streq(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2021-02-13 21:09:13 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* a_addr=local_scope[1];
|
|
|
|
nasal_val* b_addr=local_scope[2];
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-28 17:39:24 +08:00
|
|
|
ret_addr->ptr.num=(a_addr->type!=vm_str || b_addr->type!=vm_str)?0:(*a_addr->ptr.str==*b_addr->ptr.str);
|
2021-02-13 21:09:13 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_left(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2021-02-13 20:19:49 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* str_addr=local_scope[1];
|
|
|
|
nasal_val* len_addr=local_scope[2];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(str_addr->type!=vm_str)
|
2021-02-13 20:19:49 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("left","\"string\" must be string");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2021-02-13 20:19:49 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
if(len_addr->type!=vm_num)
|
2021-02-13 20:19:49 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("left","\"length\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2021-02-13 20:19:49 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
std::string& str=*str_addr->ptr.str;
|
|
|
|
int len=(int)len_addr->ptr.num;
|
2021-02-14 17:36:42 +08:00
|
|
|
if(len<0)
|
|
|
|
len=0;
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_str);
|
2021-03-28 17:39:24 +08:00
|
|
|
*ret_addr->ptr.str=str.substr(0, len);
|
2021-02-13 20:19:49 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_right(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2021-02-13 20:19:49 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* str_addr=local_scope[1];
|
|
|
|
nasal_val* len_addr=local_scope[2];
|
2021-03-28 17:39:24 +08:00
|
|
|
if(str_addr->type!=vm_str)
|
2021-02-13 20:19:49 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("right","\"string\" must be string");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2021-02-13 20:19:49 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
if(len_addr->type!=vm_num)
|
2021-02-13 20:19:49 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("right","\"length\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2021-02-13 20:19:49 +08:00
|
|
|
}
|
2021-03-28 17:39:24 +08:00
|
|
|
std::string& str=*str_addr->ptr.str;
|
|
|
|
int len=(int)len_addr->ptr.num;
|
2021-02-14 17:36:42 +08:00
|
|
|
int srclen=str.length();
|
|
|
|
if(len>srclen)
|
|
|
|
len=srclen;
|
|
|
|
if(len<0)
|
|
|
|
len=0;
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_str);
|
2021-03-28 17:39:24 +08:00
|
|
|
*ret_addr->ptr.str=str.substr(srclen-len, srclen);
|
2021-02-13 20:19:49 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_cmp(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2021-02-13 21:09:13 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* a_addr=local_scope[1];
|
|
|
|
nasal_val* b_addr=local_scope[2];
|
2021-03-27 01:08:05 +08:00
|
|
|
if(a_addr->type!=vm_str)
|
2021-02-13 21:09:13 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("cmp","\"a\" must be string");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2021-02-13 21:09:13 +08:00
|
|
|
}
|
2021-03-27 01:08:05 +08:00
|
|
|
if(b_addr->type!=vm_str)
|
2021-02-13 21:09:13 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("cmp","\"b\" must be string");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2021-02-13 21:09:13 +08:00
|
|
|
}
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_num);
|
2021-03-27 01:08:05 +08:00
|
|
|
ret_addr->ptr.num=strcmp(a_addr->ptr.str->data(),b_addr->ptr.str->data());
|
2021-02-13 21:09:13 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* builtin_chr(std::vector<nasal_val*>& local_scope,nasal_gc& gc)
|
2021-02-13 21:40:10 +08:00
|
|
|
{
|
2021-04-19 19:12:41 +08:00
|
|
|
nasal_val* code_addr=local_scope[1];
|
2021-03-27 01:08:05 +08:00
|
|
|
if(code_addr->type!=vm_num)
|
2021-02-13 21:40:10 +08:00
|
|
|
{
|
2021-03-28 17:39:24 +08:00
|
|
|
builtin_err("chr","\"code\" must be number");
|
2021-03-27 01:08:05 +08:00
|
|
|
return nullptr;
|
2021-02-13 21:40:10 +08:00
|
|
|
}
|
2021-03-30 00:12:48 +08:00
|
|
|
nasal_val* ret_addr=gc.gc_alloc(vm_str);
|
2021-03-27 01:08:05 +08:00
|
|
|
*ret_addr->ptr.str=(char)code_addr->ptr.num;
|
2021-02-13 21:40:10 +08:00
|
|
|
return ret_addr;
|
|
|
|
}
|
2021-03-27 01:08:05 +08:00
|
|
|
|
2020-09-09 17:51:31 +08:00
|
|
|
#endif
|