intel-pin

懒人玩reverse

intel-pin

之前玩了angr之后发现有点局限性,后来接触了 intel-pin 这个动态插桩工具,当时发现对于一些加密字节关联度不大的题目,尤其是代码混淆比较严重的题目,可以编写 pintool 统计指令数等信息,多快好省的通过侧信道的方法逐位爆破出 flag。

安装

官网下载即可,解压后根目录下pin程序直接可使用。

使用1

在 source/tools/ManualExamples 中有一些现成的 pintool 可以使用,基本涵盖了各个模块的用法。inscount0.cpp是指令计数插桩的功能,inscount1.cpp是基本块计数(条件分支)的插桩功能。

下面就说说指令计数的功能。

make obj-intel64/inscount0.so TARGET=intel64 编译生成64位的pintool make obj-ia32/inscount0.so TARGET=ia32 编译生成32位的pintool

pin -t your_pintool -- your_binary <arg> 使用基本命令 或者喂给输入在后面加 <<< "flag{dsada}"

通常在make前,我们可以改一改fini函数

VOID Fini(INT32 code, VOID *v)
{
    // Write to a file since cout and cerr maybe closed by the application
    // OutFile.setf(ios::showbase);
   // OutFile << "Count " << icount << endl;
   // OutFile.close();
   std::cout<< "Count " << icount << std::endl;

}

注释掉OutFile部分,不让输出文件,直接cout输出结果到命令行。

使用2

对于基本块这种情况,有些时候在程序运行过程中,我们只关心程序本身执行的基本块的个数,并不关心在外部动态链接库中的执行,因此需要使用IMG_AddInstrumentFunction记录程序镜像开始和结束地址

inscount1.cpp默认的docount函数是这样的,貌似是值记录基本块的指令数.....

我们可以定义个UINT64 bblCount = 0;全局变量,在docount函数里面添加基本块的计数:

使用3

对于一些相同的输入但是指令数不固定的程序,我们可以找到关键校验地址,对 inscount0 的 docount 函数做如下更改

需要注意的是 INS_InsertCall 是一个变参函数,前三个参数分别为指令(ins),插入的实际(IPOINT_BEFORE,表示在指令运行之前插入 docount 函数),函数指针(docount,转化为了 AFUNPTR 类型),之后的参数为传递给 docount 函数的参数,以 IARG_END 结尾,所以这里还要继续修改INS_InsertCall函数的参数。

这样再运行,只有运行到 0x47B96E 一句才会计数,这样我们就可以根据 pintool 的结果来逐位爆破 flag 了

一些脚本

猜长度

更多脚本见百度云intel-pin.

Last updated

Was this helpful?