构造使用类C语言的脚本引擎(2)作者 :kevin_qing 转贴请注明 考虑到脚本编译器部分可以单独作为一个进程实现,并且编译器需要虚拟机对其结果进行调试, 目前先从最底层的虚拟机部分开始开发。 1.虚拟机结构. 考虑到目前不支持内部函数定义,采用寄存器方式,以后扩充栈也较为方便 变量寄存器REG[256] 其中REG[0]为外部call时存放返回值 指令寄存器IP; 比较寄存器CFlag; 代码段,数据段均为虚拟指令不可改写的 1.虚拟机运行模式 使用函数数组实现,指令采取顺序编码,且按word方式对齐以加快数组索引速度 运行时函数 bool run(){ while(bRun_){ uint16_t op=op_readw(); assert((op>=0) &&(op<countof(vmop) ); vmop[op](); } } 2.指令定义 [r]代表REG[n] [i]立即数 [s]字符串 [f]外部函数ID [ip]绝对ip地址 mov 类: mov [r] [r] mov [r] [i] 运算比较指令等 重点说明函数call和switch SReset //重置选择队列 case [i] [ip]//选择id/ip对 switch [r] //开始跳转 -------------------------- FReset //重置外部函数参数队列 FPush [r] //加入一个数字参数 FPush [s] //加入一个字符串参数 FPush [i] FCall [f] //调用函数 --------------------------- switch可以不使用虚拟指令进行多次比较,而在vmop_switch()内部进行比较. 如果输入的case是排序的话,会大大节约比较时间 -------------------------------------------------------------------------- fcall的参数队列使类似printf这种可变参数得以可行,也就是说,所有外部函数都可以是可变参数。 这里是我的外部函数定义 #define SC_API __cdecl typedef int (SC_API *FP_SC_API)(int nParm,...); struct SC_API_TAB{ FP_SC_API pAPI; int nParm; const char* APIname; }; #define DECL_SCAPI(name,nParm) {&name,nParm,#name} extern const SC_API_TAB api_table[]; SC_API_TAB api_table[]={ DECL_ACAPI(say,1), ..... }; vmop_fcall的实现:采用inline汇编,因为不管是C还是C++都不能动态的实现可变参数调用。 vmop_fcall(){ int nParm=this->nParm; DWORD* pParm=fifo_parm; WORD fd=ipGetW(); void* pf=api_table[fd].pAPI; DWORD retc; __asm{ mov ecx,nParm inc ecx mov ebx,pParm mov edx,pf push_loop: dec ecx jz call_fun mov eax,[ebx] add ebx,4 push eax jmp push_loop call_fun: mov ecx,nParm inc ecx push ecx call edx mov retc,eax pop ecx dec ecx shl ecx,2 add esp,ecx } Reg[0]=retc; }
} fcall目前未做任何优化 其他指令都很容易实现,就不一一说明了。 调试: 单步执行和寄存器参看都很容易实现,也不说了. 反汇编我是通过字符串数组实现 const char* OP_[]={ 'HALT', 'MOV',//move r r 'MOV',//move r i ..... }; const uint32_t OP_SIZE[]={ 0, 2, 4, .... }; 反汇编时通过查表即可实现
|