64位PE无壳。打开main函数。
题目来源:xctf
首先判断输入长度是不是31位。不是的话就卡死。
1 | scanf("%s",input); |
根据其他师傅的wp,下面要从最后的字符对比开始分析。
1 | while( true ) { |
把DAT_140003478
的值复制出来写脚本。
1 | _Code = 0 |
得到private: char * __thiscall R0Pxx::My_Aut0_PWN(unsigned char *)
再往前:
1 | UnDecorateSymbolName(symbol_name,&out_DAT_1400056c0,0x100,0); |
关于UnDecorateSymbolName,众所周知编译器会把符号弄成人看不懂的形式(说起来以前见过一个叫mangle的词)(千万不要用ecosia搜mangled这个词,会出血腥图,呕呕呕)(啊,这个是gcc和clang的东西,见http://web.mit.edu/tibbetts/Public/inside-c/www/mangling.html)
扯远了,vs的规则可以在https://zh.wikipedia.org/wiki/Visual_C%2B%2B名字修饰 查看。但是更方便的方法是直接把这个函数写出来,然后用__FUNCDNAME__
宏输出。
1 |
|
得到?My_Aut0_PWN@R0Pxx@@AAEPADPAE@Z
另外插一句嘴,一定要用能分得清0和O的字体。cmd的字体可以设为点阵字体(不知道为什么vs的输出窗口字体只有几种能选的,如果设成别的就会找不到字体然后掉回宋体)。
最后我们剩下的就是这一段了:
1 | iVar2 = FUN_140001280(input); |
省流:动态调试可以发现是一个位置上的置换,按它的方式换回去就可以了。(但我还是分析了一下这个二叉树)
Node*
类型是我们定义的结构体(为什么是‘们’呢因为ghidra的自动生成帮了一部分忙)。
Offset | Length | Type | Name |
---|---|---|---|
0 | 1 | char | v |
1~7 | 1 | undefined | |
8 | 8 | Node * | left |
16 | 8 | Node * | right |
然后是dfs函数,在把它的名字改成dfs之前,我还看不懂它。
不过现在很明显了,是一个后序遍历,存到sym_name
里面
1 | void dfs(Node *param_1) |
用ida调试查看一下root的结构,看看树建成了什么样:
(ida可能无法识别把一个内存中的变量分到两个寄存器这个操作,比如这里需要在v6上右键,映射到v4。)
这个叫什么二叉树来着我给忘了,反正就是按层的顺序,从左到右。
顺便练一下用f#写数据结构。
先深搜把数据填进去,然后广搜还原建树的顺序。
1 | open System.Collections.Generic |
得到:Z0@tRAEyuP@xAAA?M_A0_WNPx@@EPDP
最后md5一下就好了。
碎碎念:用尝试f#写二叉树的时候我最开始想把结点搞成可变的。结果就变成了type Node ={ v: char;l: Node ref option;r: Node ref option}
想取到里面的Node
?先解两层包。然后引用传参最开始写的是Node ref类型,写完给我个蓝色warning,然后才知道现在都是写byref。然后返回的时候还不能直接取成员地址,让我先let绑定。
然后我就崩溃了,一查才知道原来是要修改结点的时候直接扔掉返回一个新的。啊,我忘了该用函数式写法了。(然而我还是用了一个全局变量计数。)
其实这道题是我随机到的,就是感觉按难度挨着做,中间突然冒出几道新题,无法满足强迫症的感觉,干脆从这里打破吧。然后随机到这个难度6的题。然后点进建树的函数被吓出来。然后开摆看wp。然后告诉我这是签到题。呜呜呜。