asmjit是一个开源项目,使用它可以将代码即时的编译成机器码,也就是所谓的jit技术。
初次接触这个项目,编写了一个demo,学习它的使用方法。
现将编写的demo以及调试jit生成的机器码的过程总结出来,分享给大家
asmjit调用自定义方法的demo
代码如下
#include "asmjit.h" // [Dependencies - C] #include <stdio.h> #include <stdlib.h> #include <string.h> using namespace AsmJit; #include <string> #include <iostream> using namespace std; int add(int value1, int value2) { cout << "arg1: " << value1 << " arg2: " << value2 << endl; return value1 + value2; } int main(int argc, char* argv[]) { X86Compiler c; //记录日志,将编译细节输出到控制台 FileLogger Logger(stdout); c.setLogger(&Logger); //新建一个无参数、无返回值的方法 c.newFunc(kX86FuncConvDefault, FuncBuilder0<void>()); //定义一个临时变量 GpVar result(c.newGpVar()); //给add方法传入参数 c.push(Imm(9)); c.push(Imm(10)); //调用自定义的add方法 c.call((void*)add); //方法结束 c.endFunc(); typedef void (*myfun)(void); //生成方法的机器码--真正的机器码 myfun fun = asmjit_cast<myfun>(c.make()); //调用刚刚生成的方法 fun(); //获取add方法的返回值 __asm{ mov value3, eax } cout << value3 << endl; //程序结束释放生成的方法占用的内存 MemoryManager::getGlobal()->free(fun); return 0; }
VS调试
用VS调试生成的fun方法,首先通过调试菜单,打开反汇编窗口,如下图
通过F11单步调试进生成的fun方法,看到如下的反汇编代码
由于显示一堆的??,无法知道生成的是什么指令,所以无法调试。
针对此问题在asmjit的google groups进行了提问
下面是比较有用的回答
根据Palo Marton的回答,这应该是VS的BUG。
windbg调试asmjit生成的方法
既然VS2012有BUG,那么就用windbg来调试吧
启动后通过 lm 指令查看当前加载的模块
上图看到,asmjit的符号文件还未加载。
对main方法设置断点 bp asmjit!main + 10,然后通过bl查看当前断点
查看当前模块及其符号文件
可以看到,这时asmjit的符号文件已经加载了。
通过g命令直接运行。
此时由于符号文件已经加载,windbg会根据符号文件加载对应的源文件
这时断在了刚才下的断点处,Alt+7打开反汇编窗口
通过p命令进行单步调试(相当于VS中的F10),t命令进行步入调试(相当于VS中的F11)。
执行到fun方法时,通过t命令进入方法内就可以看到生成的fun方法的反汇编代码了
调试过程中使用到的windbg命令介绍
lm:List Loaded Modules,lm 命令显示已经加载的模块,输出信息包括模块的状态和路径。
bp:bp命令设置程序断点。
bl:BreakPoint List ,bl命令列出当前存在的断点信息列表。
p:Step,p命令执行一条指令或者一句源代码,然后展示所有寄存器的结果,相当于VS调试中的F10。
g:Go,g命令开始执行当前进程或线程,相当于VS中的F5。
t:Trace,t命令相当于VS中的F11。
gu:Go Up,gu命令会使程序一直执行,知道当前的方法执行结束。相当于VS中的Shift+F11。