漏洞类型及攻略方法
全写出来不太可能,图示大概这样:
data:image/s3,"s3://crabby-images/5e92f/5e92fbdb0c17bbaae9e836b1c9e6093f7672e31f" alt=""
Stack
以x86环境为例
每个进程在内存区域底部都有一个stack
data:image/s3,"s3://crabby-images/45b85/45b85ee675554e29461585acdef580283deaea8f" alt=""
- 上半部分是在32位机器中运行32位程序
- 下半部分是在64位机器中运行32位程序
它与C语言函数密切相关
1 | int main(void){ |
例如,调用这个函数时stack状态如下:
data:image/s3,"s3://crabby-images/d980c/d980c18e63d249b5b439aac58dc2c9b63c1c5d38" alt=""
ESP/EBP不容易理解,这里解释一下
data:image/s3,"s3://crabby-images/74d44/74d444ff0b6599dee4aa249efc7dd937b2c1ed18" alt=""
- ESP寄存器始终指向栈顶
- EBP寄存器始终指向栈底
data:image/s3,"s3://crabby-images/a4f66/a4f663c69e0a73d79130e4620e035a128a71f28e" alt=""
- EBP-0x8能偶访问本地变量2
- EBP-0x4能够访问本地变量1
- EBP+0x8能够访问参数1
- EBP+0xC能够访问参数2
data:image/s3,"s3://crabby-images/0e895/0e895360e5a4fd66e4a7ab96b16ba4a78b94fab0" alt=""
调用原函数也有相同的帧并保存EBP寄存器,当函数被调用时,EBP会以新的帧为基础更新。因此,调用原函数的帧暂时保存到stack中
栈溢出
对策及解决方法
首先来看经典问题
曾经,stack和heap都是有X权限的
BOF:buffer overflow
写入数据超出区域
覆盖写入后续区域
例如这段代码:
int main(void){ int a, b; .. func(a, b); .. return 0; } void func(int a, int b){ char buf1[4]; char buf2[4]; strcpy(buf2,"AAAABBBB"); }
int main(void){ int a; char s[256]; .. read(stdin, s, 256); func(a, s); .. return 0; } void func(int a, char* s){ char buf1[4]; char buf2[4]; strcpy(buf2, s); }1
2
3
4
5
6
7
8
9
10
11
12
data:image/s3,"s3://crabby-images/c86d0/c86d0ccc1509844b4fc0707687c1806b15713b2c" alt=""
- 如果用户可以输入任意值,会发生什么?
- 如果没有size限制,就可以任意进行覆盖
- 甚至可以指定返回地址
- 例如这段代码:
-push byte 0x3 ; eax = NR_read pop eax push byte 0x4 ; ebx = fd pop ebx push byte 0xff; edx = size pop edx call next next: pop ecx ; ecx = buf(=ret_addr) int 0x80 ; read(fd, buf, size)1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
data:image/s3,"s3://crabby-images/609b9/609b96fa9c38fb8a70adb7525c38a3e53f059084" alt=""
- 添加shellcode
- 如果Evil Return addr指向shellcode
- 那么返回前会执行shellcode并执行shell(/bin/sh)
- 可以执行任意代码
data:image/s3,"s3://crabby-images/2fe97/2fe9743f03fde788d7ec36b31a4654ecbcda0073" alt=""
## 1-猜测地址
- stack地址是动态的,取决于环境
- 如何让它到达shellcode
data:image/s3,"s3://crabby-images/4fe7a/4fe7a7e16812225b1280f4549ed490600caa5892" alt=""
- 可以略微转移
- nop-sled
- nop = 0x90(空指令)
- 尽可能多的拼接
- 如果到达nop-sled的某个地方,shellcode将自动执行
- 多次尝试
- brute force
- 如果不知道地址,那么可以尝试所有的可能
- 结合使用nop-sled可以减少尝试次数
data:image/s3,"s3://crabby-images/f9f4e/f9f4e1ab9269e3f77b9ce939d4a6b1ad3b3c9bfd" alt=""
## 2-溢出量的问题
- 考虑不能过多溢出的情况
- nop-sled无法准备好(或者太短了)
- 不能猜测地址
- brute force需要太长时间
- 如果运气好的话,能够执行shellcode
- 实践中很少会成功
data:image/s3,"s3://crabby-images/3280c/3280cd053f77406da31733a3ba0deef635e769db" alt=""
- 使用保存栈地址的ESP比较好
- ret2esp
- 但是,如果代码中没有jmp esp或者call esp就很糟糕
- 实际上很可能找不到
data:image/s3,"s3://crabby-images/7f47a/7f47ae68fb9f66daf5aea18057eb3ee48143c378" alt=""
## 3-溢出量的问题
- 只能够少量溢出的情况
- nop-sled无法构造
- 甚至连填充shellcode都不够
data:image/s3,"s3://crabby-images/c126a/c126aede7a7120c88f183a725abf73e481fedc99" alt=""
- 可以使用较短的代码来读取后面更多内容
- Stager
- 通过溢出写入较短的汇编代码
- 先运行这部分代码,读取其他shellcode
- 通过读取的内容来控制程序运行流程
- 具体大概像这样( 只有17个byte):
-stager的优点
- 使用Shellcode时,目的是首先执行Stager
- 如果Stager移动,Shellcode也会移动
- 因为它很短,很容易绕过限制
- 可以轻松更改后续shellcode
对策 - Stack不可执行(NX)
如果stack和heap不能执行就太好了
这就是NX(No eXecute / Never eXecute)
有各种实现,例如NX比特位,PaX,Exec Shield(Redhat)等
- 硬件实现或软件实现
结果都是让stack之类不可执行
考虑绕过NX
- 该怎么做
- (例)创建可写的固定地址,编写shellcode并执行
- 通过mmap或者mprotect获取RWX权限的内存区域
- 在那里写入shellcode并返回
问题点
- 如何调用到mmap/mprotect?
- 通过溢出如何写入到那里
- 溢出是在栈向下写入的攻击
- 不能够反向写入
data:image/s3,"s3://crabby-images/d4974/d4974ea7b590667df9714787d2d63ab127a3ae3e" alt=""
ret2plt
- 让栈的状态,在函数调用的瞬间是相同的(非常重要的概念)