首页
社区
课程
招聘
2
[原创] KCTF 2019 Q3 第十三题 Writeup by Nu1L & 没做出来的题目的分析
发表于: 2019-9-22 23:04 3404

[原创] KCTF 2019 Q3 第十三题 Writeup by Nu1L & 没做出来的题目的分析

2019-9-22 23:04
3404

check在 0x407f30
看到了熟悉的大整数计算
拆 partA KXCTFXXXX partB

partB 十六进制数 0-9A-F 长度大于8
partA 就是username这个字符串
partB + cdeae8b5b9e9d5d4 + deadbeefcafec0de + cdeae8b5b9e9d5d4cdeae8b5b9e9d5d4做了一个sha1
这里的+是连接

sha1字符串后面又放了一个KXCTF19Q3

对几个东西都sha1了一下,都是固定的,可以提出来

其中下标为1的是username这个字符串的sha1,看出来作者是想做一队一密的
还有几个常数,提取一下

逆向过程有点痛苦
整理一下算法
其中有一个F函数比较迷,单独拿出来看

整体流程大概这样


翻了一整篇gmp的文档,经过一些运算性质的判定和测试,确定F函数求的是Lucas序列
https://en.wikipedia.org/wiki/Lucas_sequence
返回值是(Ux, Vx, V(x-1))

然后要处理一下,化简一下这个里面的那些神奇的运算
现在已经知道

其中P和Q是Lucas序列的参数,k是需要求第k项,n是模数,U(k)和V(k)是Lucas U序列和V序列的第k项
首先,因为在模n的有限域里面,所以所有大于n的数都可以等价为模n

显然x等价于1
设flag=e,等价于flag_num+sha1s[0]
那么后面涉及的Lucas序列就都是P=flag Q=1的
F函数操作一下之后,获得了两个K=shas的序列值
sha1(flag) = 0xSSSSSSSSSSSSSSSSSSSSSSSSSSSSSSSSSSSSSSSS
设取反是0xRRRRRRRRRRRRRRRRRRRRRRRRRRRRRRRRRRRRRRRR
显然对于每一个十六进制位,Si+Ri=0xF
那么得到了k1,k2是

那么xx,yy可以得到

下面看这一波运算

ps数组里面都是1,暂时不管
看一下前两个式子
U序列和V序列都有一些运算性质的
根据性质

可得第一个式子在算U(2k)
然后根据

这里Q刚好是个1,所以就是第二个式子
所以这一个循环就是把U和V的k值乘了2
一共循环了72次,就是乘了2的72次方,也就是左移72位,就是9个字节
跑完了以后,现在k1和k2相当于

看上去有趣了起来
然后继续看下面的运算

根据之前的式子可以看出来f是一个模d的除2操作
所以实际上是

然后Lucas序列又有如下性质

其中

显然,就是上面那个式子的形式
那么这时候z就是V(k1+k2)
k1和k2之前知道了

然后又因为两个sha1是按位取反的关系,所以任意Si+Ri=0xF
这一加起来就是

SHA1消掉了!
算了这么多,到现在就是一个式子

继续往下算
给z加了一个sha1s[1]

这里x=d+1,所以w可以算,也是d+1
然后算了P=z Q=1 模数为d的lucas序列的第w-1项,就是第d项
实际上就是

最后加上一个sha1s[3],模d看是否等于sha1s[2]

所以总结一下,流程是这样

可以看到实际上就是算了两个式子,真实
那么问题就转化为
已知Q=1,k,n,以及V(k),求P

是Q=1的Lucas序列,可能有一些特殊性质,去找一找资料

读论文

这个是一种基于Lucas序列的公钥加密体制
完全分解N的情况下可以求私钥

这个题N不大,可以分解
逆!

32字符flag 字母数字
读进去转成64进制数字
作为opcode

整理一下VM功能表

不断的循环跑这32字节的bytecode,直到r0=128算跑一遍
一共跑128遍

这能逆?

猜一下,源数组和目标数组里面都是0-127全来了一遍,一般的运算凑不出来,可能是代换/置换盒,这几个运算里面能简单做代换的就异或
14524627489 4624527489 452462749 00
但是好像不对
作为置换盒的话,连续置换几次是会回去的,感觉可能是个突破口

带混淆的安卓
https://bbs.pediy.com/thread-253533.htm
https://github.com/GeT1t/deollvm/blob/master/cfg.py

是控制流平坦化+指令替换+虚假控制流

发现了NISLFILE,根据特征找到这个文章

两个virtualprotect,更改前后如下

文件头被改了,文件头:
图片描述
事先把保留字给改成了一个地址,然后执行fld,将这个地址放入st0,不知道在干什么,先拿出来备用

又把这玩意拿出来了,stackbase和stacklimit,两个地址,做差,算了一个size,地址以4(Dword对齐),放回dos头里,就是算了线程栈的大小然后把线程栈里面的东西复制到wtf中,备份esp,ebp.

前面就是因为start没什么内存,先是用没用的dos头存数据,然后把栈迁移到另外的一个地址,备份esp,ebp,开始脱壳.试一下esp定律,下了个硬件断点,好像ida炸了

过了一个指令异常,前一堆rdtsc,后面下断,找到一个call eax,东西在这里面
图片描述
有地址随机化

跟进eax,还是壳的部分,按照脱壳的套路来,有上跳就下断,然后好几处call eax,估计在循环解码,有常量加密

一次567000一次其他的地址,运算全部带有常量混淆,以及所有的指令都是jmp这种,正好和控制流平坦化相反了,这玩意流程是一个纵向的

一堆rdtsc好像没什么影响,OD可以直接过反调试,rdtsc对OD无效
下了个puts(好像是)的断点,在最后打印的时候,看到了结尾左右的逻辑,popad后进行判断然后跳转,最后打印后执行Terminal

调吧 = =,好像并没有发现什么指令集的转换,都是一些很无聊的跳转加上循环


[培训]内核驱动高级班,冲击BAT一流互联网大厂工作,每周日13:00-18:00直播授课

最后于 2019-9-25 22:06 被acdxvfsvd编辑 ,原因:
收藏
免费 2
支持
分享
赞赏记录
参与人
雪币
留言
时间
PLEBFE
为你点赞~
2023-1-22 04:24
brucy
为你点赞~
2019-9-25 15:27
最新回复 (0)
游客
登录 | 注册 方可回帖
返回

账号登录
验证码登录

忘记密码?
没有账号?立即免费注册