JIT动态编译器的原理与实现之Interpreter3
接下来,就是要实现一个虚拟机了。记得编码高质量的代码中有一条:不要过早地优化你的代码。所以,也本着循序渐进的原则,我将从实现一个解释器开始,逐步过渡到JIT动态编译器,这样的演化可以使原理看起来更清晰。
解释器的原理很简单,就是一条指令一条指令的解释并执行。具体流程分为:取出指令-解码指令-执行-返回主流程。这样形成一个无限循环,如下图所示:
这里的主流程就是上篇定义的程序rom.bin。但rom.bin不能直接运行,需要一个解释器来包裹它,来解释执行。解释器放在一个无限循环中,使得主流程无限运行不停止:
void loop() { for (;;) { Interpreter(&CPUREG); } } |
这样,整个虚拟机的运行可以定义为:
memInit(); //初始化内存 ResetCPU(&CPUREG); //初始化CPU loadROM(); //加载rom.bin loop(); //执行主流程 memFree(); //释放内存 |
接下来需要做的就是取出指令送入解释器了。为此需要定义读写内存的函数memGet和memSet:
void memSet(unsigned int , unsigned char ); unsigned char memGet(unsigned int ); void memSet(unsigned int addr, unsigned char data) { char Str_Err[256]; if (addr>64) { sprintf (Str_Err, "MEM: invalid mem write: 0x%8x" , addr); MessageBox(NULL, Str_Err, "Warning" , MB_OK); } else { RAM[addr & 0xff]=data; } } unsigned char memGet(unsigned int addr) { char Str_Err[256]; unsigned char val = 0; if (addr>64) { sprintf (Str_Err, "MEM: invalid mem read: 0x%8x" , addr); MessageBox(NULL, Str_Err, "Warning" , MB_OK); } else { val=RAM[addr & 0xff]; } return val; } |
读写均为一个字节。由于上篇定义的CPU寻址范围只有64字节大小,所以超过64字节就要给出错误提示。
然后需要为每一个CPU指令机器码实现一个解码执行函数:
void nop(REG*); void mov(REG*); void add(REG*); void cmp(REG*); void jmp(REG*); void jcp(REG*); void nop(REG* cpuREG) { cpuREG->R_PC++; sprintf ( "NOP\n" ); } void mov(REG* cpuREG) { memSet(cpuREG->R_PC+1, memGet(cpuREG->R_PC+2)); sprintf ( "MOV [0x%4x], [0x%4x]\n" , cpuREG->R_PC+1, cpuREG->R_PC+2); cpuREG->R_PC+=3; } void add(REG* cpuREG) { memSet(cpuREG->R_PC+1, memGet(cpuREG->R_PC+1)+memGet(cpuREG->R_PC+2)); sprintf ( "ADD [0x%4x], [0x%4x]\n" , cpuREG->R_PC+1, cpuREG->R_PC+2); cpuREG->R_PC+=3; } void cmp(REG* cpuREG) { if ((memGet(cpuREG->R_PC+1)-memGet(cpuREG->R_PC+2)) < 0) { cpuREG->R_CMP=0; } else { cpuREG->R_CMP=1; } sprintf ( "CMP [0x%4x], [0x%4x]\n" , cpuREG->R_PC+1, cpuREG->R_PC+2); cpuREG->R_PC+=3; } void jmp(REG* cpuREG) { sprintf ( "JMP [0x%4x] \n" , cpuREG->R_PC+1); cpuREG->R_PC=memGet(cpuREG->R_PC+1); } void jcp(REG* cpuREG) { sprintf ( "JCP [0x%4x], [0x%4x]\n" , cpuREG->R_PC+1, cpuREG->R_PC+2); if (cpuREG->R_CMP==0) { cpuREG->R_PC=memGet(cpuREG->R_PC+1); } else { cpuREG->R_PC=memGet(cpuREG->R_PC+2); } } |
这里最重要的是要小心处理PC寄存器。一开始CPU初始化的时候,PC寄存器是设为0的,而自定义的rom.bin也是从0地址开始执行的。如果你虚拟的CPU不是从0地址开始执行,那么在CPU初始化的时候就要把PC寄存器设为相应的开始地址。另外每一条指令可能涉及的地址数不相同,那么PC寄存器的变动也要不同。最后,跳转指令也可能要根据比较寄存器的内容来改变PC寄存器。
做了如上的准备之后就可以实现解释器了。这里用switch-case结构来决定哪条指令被执行。为了简单起见,用了一个函数指针来执行解码函数:
void (*func)(REG*); //Interpreter void Interpreter(REG* cpuREG) { char Str_Err[256]; switch (memGet(cpuREG->R_PC)) { case 0: func=nop; break ; case 1: func=mov; break ; case 2: func=add; break ; case 3: func=cmp; break ; case 4: func=jmp; break ; case 5: func=jcp; break ; default : sprintf (Str_Err, "Unhandled Opcode (0x%4x) at [0x%4x]" , memGet(cpuREG->R_PC), cpuREG->R_PC); MessageBox(NULL, Str_Err, "Warning" , MB_OK); return ; } func(cpuREG); } |
首先从内存中取出数据,根据机器码来决定执行解码函数,最后执行。执行结果如下:
JIT动态编译器的原理与实现之Interpreter3的更多相关文章
- JIT动态编译器的原理与实现之Interpreter(解释器)的实现(三)
接下来,就是要实现一个虚拟机了.记得编码高质量的代码中有一条:不要过早地优化你的代码.所以,也本着循序渐进的原则,我将从实现一个解释器开始,逐步过渡到JIT动态编译器,这样的演化可以使原理看起来更清晰 ...
- gcc/g++等编译器 编译原理: 预处理,编译,汇编,链接各步骤详解
摘自http://blog.csdn.net/elfprincexu/article/details/45043971 gcc/g++等编译器 编译原理: 预处理,编译,汇编,链接各步骤详解 C和C+ ...
- java9新特性-21-java的动态编译器
1. 官方Feature 243: Java-Level JVM Compiler Interface 295: Ahead-of-Time Compilation 2. 产生背景 Oracle 一直 ...
- Java动态编译技术原理
这里介绍Java动态编译技术原理! 编译,一般来说就是将源代码转换成机器码的过程,比如在C语言中中,将C语言源代码编译成a.out,,但是在Java中的理解可能有点不同,编译指的是将java 源代码转 ...
- Java 动态调试技术原理及实践
本文转载自Java 动态调试技术原理及实践 导语 断点调试是我们最常使用的调试手段,它可以获取到方法执行过程中的变量信息,并可以观察到方法的执行路径.但断点调试会在断点位置停顿,使得整个应用停止响应. ...
- Android JIT实时编译器的设置
在Android JIT实时编译是在Android 2.2之后才引入的,JIT编译器可以显著的提高机器的性能,经过测试,android 2.2的性能较android 2.1提高了 2-5倍.JIT提 ...
- java高级---->Java动态代理的原理
Java动态代理机制的出现,使得 Java 开发人员不用手工编写代理类,只要简单地指定一组接口及委托类对象,便能动态地获得代理类.代理类会负责将所有的方法调用分派到委托对象上反射执行,在分派执行的过程 ...
- Cglib动态代理实现原理
Cglib动态代理实现方式 我们先通过一个demo看一下Cglib是如何实现动态代理的. 首先定义个服务类,有两个方法并且其中一个方法用final来修饰. public class PersonSer ...
- RxJava RxPermissions 动态权限 简介 原理 案例 MD
Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱 MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina ...
随机推荐
- aspnet-webapi-2-contrib
https://github.com/rdingwall/protobuf-net-data https://github.com/mgravell/protobuf-net https://gith ...
- UIAutomator源码分析之启动和运行
通过上一篇<Android4.3引入的UiAutomation新框架官方简介>我们可以看到UiAutomator其实就是使用了UiAutomation这个新框架,通过调用Accessibi ...
- C# ASP.NET CSV文件导入数据库
原文:C# ASP.NET CSV文件导入数据库 using System; using System.Collections.Generic; using System.Text; using Sy ...
- PS抠出树叶树枝
1.打开PS 2.加载树叶树枝图片 3.双击该图层,来解锁树叶树枝图层 4.通道面板,只留下蓝色 5.顶部菜单 -> 图像 -> 计算 -> 混合为正片叠底,得到一个新Alpha图层 ...
- 收集整理的非常有用的PHP函数
原文:收集整理的非常有用的PHP函数 项目中经常会需要一些让人头疼的函数,作为开发者应该整理一个自己的函数库,在需要之时复制过来即可.本文作者收集整理数十个PHP项目中常用的函数,保证能正常运行,你只 ...
- 让IE6支持position:fixed的方法,CSS expression与JavaScript eval讲解
做吸顶效果或是固定效果时,使用position:fixed无非是最方便的,可是万恶的IE6是没有fixed这个属性值的,而我们要使IE6能够像fixed一样固定在浏览器中的某个位置,使用onscrol ...
- vector成员函数解析
vector线性集装箱,其元素颜格排序根据线性序列,和动态数组很阶段似,像阵列,它的元素被存储在连续的存储空间,这也意味着,我们不仅能够使用迭代器(iterator)访问元素,也可以用一个指针访问偏移 ...
- 雕爷:我眼中的O2O成长路径
嗨,老周,这篇文章写给你.知道你最近正在纠结于O2O的择业分析,因为你是有极高身价的人,所以选择起来必须谨慎,选错了,不是身家那千把万的损失,更是一生荣耀的赌注和起落. “所谓战略,就是站在未来看今天 ...
- 个推推送 产品SDK常见问题检查
作者:Hong Jack链接:https://zhuanlan.zhihu.com/p/20733333来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处. 通知和消息有什么 ...
- 推送之HelloWorld及个推Smart Push
最近有个朋友想要推送一些消息到自己的APP上,自己用了HTTP轮询的方式比较耗电,也比较占用流量,一旦用户关闭了进程,消息则很难触达,于是,咨询我有没有什么好的解决方案.我告诉他其实可以使用推送,他瞪 ...