|
第二十五章------溢出 " \! B: B- [ }( y" U
简单说明: c& j" {" ]) g( y
溢出对我们这些小鸟们属于比较高级的玩意了。所以在这里也不准备多提,否则就违反了这篇文章写给初学者的初衷了。
1 b! i- v) g! }1 R. p1 a缓冲区溢出
" F. m6 i, @# T \% i5 G缓冲区是内存中存放数据的地方。在程序试图将数据放到计算机内存中的某一位置,但没有足够空间时会发生缓冲区溢出。
% v+ @: J9 q- L$ X1 |缓冲区是程序运行时计算机内存中的一个连续的块,它保存了给定类型的数据。问题随着动态分配变量而出现。为了不用太多的内存,一个有动态分配变量的程序在程序运行时才决定给他们分配多少内存。如果程序在动态分配缓冲区放入太多的数据会有什么现象?它溢出了,漏到了别的地方。一个缓冲区溢出应用程序使用这个溢出的数据将汇编语言代码放到计算机的内存中,通常是产生root权限的地方。单单的缓冲区溢出,并不会产生安全问题。只有将溢出送到能够以root权限运行命令的区域才行。这样,一个缓冲区利用程序将能运行的指令放在了有root权限的内存中,从而一旦运行这些指令,就是以root权限控制了计算机。总结一下上面的描述。缓冲区溢出指的是一种系统攻击的手段,通过往程序的缓冲区写超出其长度的内容,造成缓冲区的溢出,从而破坏程序的堆栈,使程序转而执行其它指令,以达到攻击的目的。据统计,通过缓冲区溢出进行的攻击占所有系统攻击总数的80%以上。造成缓冲区溢出的原因是程序中没有仔细检查用户输入的参数。例如下面程序:?/P> k. b& ^6 x+ C6 ]
example0.c2 K6 |0 V) M d7 C
-----------------------------------------------------------
V1 I. F1 Q8 a' ~* R$ \ ^ void function(char *str) {, h& R& v" e: a# K1 |5 f/ |
char buffer[16];
; d% i2 _9 u6 [7 \. T$ c% f; f
( ?0 Q7 q |: N8 d2 b( U7 E$ ?" B k f strcpy(buffer,str);, h+ T* H1 Z4 ^6 `+ h3 T
}
% J& u3 |5 G X) V0 @- B ----------------------------------------------------------- 7 P: T& w% y! v4 y* h6 C- ~
上面的strcpy()将直接把str中的内容copy到buffer中。这样只要str的长度大于16,就会造成buffer的溢出,使程序运行出错。存在象strcpy这样的问题的标准函数还有strcat(),sprintf(),vsprintf(),gets(),scanf(),以及在循环内的getc(),fgetc(),getchar()等。在C语言中,静态变量是分配在数据段中的,动态变量是分配在堆栈段的。缓冲区溢出是利用堆栈段的溢出的。一个程序在内存中通常分为程序段,数据端和堆栈三部分。程序段里放着程序的机器码和只读数据,这个段通常是只读,对它的写操作是非法的。数据段放的是程序中的静态数据。动态数据则通过堆栈来存放。在内存中,它们的位置如下: 1 ?6 A6 A0 h1 A7 R" E
2 v5 D5 {# C$ B9 N ?! x6 n: v$ S /――――――――\ 内存低端
; J9 Q* |- L& E+ f6 A! ? |程序段|
8 i1 ~% Q+ }% Q+ N/ C$ d |―――――――――|
% v2 c/ N3 z4 {" L d | |数据段|
7 a3 h6 J9 K7 T! k; W. h4 Z |―――――――――|
+ I, J% ]! Y* U( Q; @ |堆栈|- i+ m7 \, B0 H9 E
\―――――――――/内存高端 1 J @# \* {$ Y+ ^
堆栈是内存中的一个连续的块。一个叫堆栈指针的寄存器(SP)指向堆栈的栈顶。堆栈的底部是一个固定地址。堆栈有一个特点就是,后进先出。也就是说,后放入的数据第一个取出。它支持两个操作,PUSH和POP。PUSH是将数据放到栈的顶端,POP是将栈顶的数据取出。在高级语言中,程序函数调用和函数中的临时变量都用到堆栈。参数的传递和返回值是也用到了堆栈。通常对局部变量的引用是通过给出它们对SP的偏移量来实现的。另外还有一个基址指针(FP,在Intel芯片中是BP),许多编译器实际上是用它来引用本地变量和参数的。通常,参数的相对FP的偏移是正的,局部变量是负的。当程序中发生函数调用时,计算机做如下操作:首先把参数压入堆栈;然后保存指令寄存器(IP)中的内容,做为返回地址(RET);第三个放入堆栈的是基址寄存器(FP);然后把当前的栈指针(SP)拷贝到FP,做为新的基地址;最后为本地变量留出一定空间,把SP减去适当的数值。 # R# L z2 e1 }' V* f- s$ r
下面举个例子:
3 e8 d( H% n/ Y/ X# u example1.c:
' U# E% n$ [& x4 Y: @+ M1 b6 x7 ^( H. G ------------------------------------------------------------5 c, C3 k2 @6 o) k- {; Q" x
void function(int a, int b, int c) {& |6 {2 f% f1 G4 X4 O2 k
char buffer1[5];
1 m- B9 s: T' Q( v! w& m char buffer2[10];
$ p# \% s/ V w/ ]9 g* j }+ c6 ^( x8 w' O) ^1 {8 O3 t
7 {9 {! c* O$ c* G8 P" h void main() {
: L. e- f$ Z: h1 L function(1,2,3);( H2 r$ u9 v9 ]9 S0 V% b* _
}1 L4 J8 {$ B) h# \" a, l* y
-----------------------------------------------------------
8 T+ V: q( _0 J4 [2 p为了理解程序是怎样调用函数function()的,使用-S选项,在Linux下,用gcc进行编译,产生汇编代码输出: 3 M/ N% E0 O. l) U! t. J" X: @
$ gcc -S -o example1.s example1.c
- |, x: a1 x* n/ X5 d 看看输出文件中调用函数的那部分: . \0 A$ C! c1 x+ l( D1 y0 R
pushl $37 G" F( j! C) k! D9 _
pushl $2
* v* P. r* H% b- [/ l% l K pushl $1$ L3 L4 r, G# V& E; w* o0 W( z# u
call function
, x' ?; F a; y" L) }2 |% q 这就将3个参数压到堆栈里了,并调用function()。指令call会将指令指针IP压入堆栈。在返回时,RET要用到这个保存的IP。在函数中,第一要做的事是进行一些必要的处理。每个函数都必须有这些过程:
- g* v, i: S% v6 r) R1 {
$ u9 ^/ b, j% L& K3 s8 M pushl %ebp
. o* l9 [1 {/ H movl %esp,%ebp
# u: i) w( Z: Y ] subl $20,%esp
+ a2 B7 r4 t7 \: ?: q L2 R这几条指令将EBP,基址指针放入堆栈。然后将当前SP拷贝到EBP。然后,为本地变量分配空间,并将它们的大小从SP里减掉。由于内存分配是以字为单位的,因此,这里的buffer1用了8字节(2个字,一个字4字节)。Buffer2用了12字节(3个字)。所以这里将ESP减了20。这样,现在,堆栈看起来应该是这样的。
6 {+ c& P1 N; [( v6 C
r- `3 Y1 i2 }! U 低端内存高端内存5 S) M; D8 k" X3 @9 R( E
buffer2 buffer1 sfp ret a b c
& p8 V: [5 `, x < ------ [ ][ ][ ][ ][ ][ ][ ]# X$ u) a+ |' y4 b) h+ p- a4 e
栈顶栈底
* }: G% a _3 V3 b+ L1 [
' h/ Y5 ]$ p$ j' w- r+ | 缓冲区溢出就是在一个缓冲区里写入过多的数据。那怎样利用呢,看
j& z R4 R- }+ u一下下面程序:
9 A9 X) _- Q0 }' x
- \9 r: K! B( A* i! L
; H3 I& e) l. L' J+ W7 e2 p7 G example2.c" m% y& w5 K# S& K- g
-----------------------------------------------------------7 ~' B$ \0 `: W" u5 ]
void function(char *str) {
4 o7 A% u" Y* t2 W5 N+ Q& n# o char buffer[16];
' A' ]+ U9 U* F# X3 z. J/ r: S 9 ?/ a9 r+ S5 J# @
strcpy(buffer,str);0 h1 j/ g0 U9 Q* ]+ _% X
}1 O: t9 N2 W- Q$ J
' u% R0 m& H, K) h void main() {9 Q: I4 G% B# k) O5 Z
char large_string[256];
% ]4 S. D( F5 }! a, w int i;
# B* V9 ?$ |3 }* x$ O) D2 Z5 f# r 0 a9 F+ c1 P# C7 {2 @6 A" p
for( i = 0; i < 255; i++)8 Q2 l. d- e T2 {
large_string = A;! e& x F( ?2 B) E2 U0 G
. p1 ?! L4 h! e. d2 `. m, d
function(large_string);
# H; \, `! [. ]: e }
6 g$ r2 u' ^, z- f' n* o; ? ------------------------------------------------------------ 7 L% K' r5 H5 O, \9 i
这个程序是一个经典的缓冲区溢出编码错误。函数将一个字符串不经过边界检查,拷贝到另一内存区域。当调用函数function()时,堆栈如下:
' x* [5 J, {7 n, R0 z8 D- y ! D$ t2 [6 G. X* R/ h4 h* T
低内存端buffer sfp ret *str高内存端
" V" C; B2 \+ u. |. k1 D < ------ [ ][ ][ ][ ]) l5 B. o$ ^7 F7 h+ L, e
栈顶栈底 / ]4 P; ]2 _( ^# s$ Y# H
很明显,程序执行的结果是"Segmentation fault (core dumped)"或类似的出错信息。因为从buffer开始的256个字节都将被*str的内容A覆盖,包括sfp, ret,甚至*str。A的十六进值为0x41,所以函数的返回地址变成了0x41414141, 这超出了程序的地址空间,所以出现段错误。可见,缓冲区溢出允许我们改变一个函数的返回地址。通过这种方式,可以改变程序的执行顺序。
6 q) @. Z {; a; N: u* ^由于是简单介绍给大家熟悉一下,所以在这里不给出软件了,如果大家有需要的话,可以在附录中找到。
8 P8 H2 l% H: X相关资料:
' g, l; V& P$ P- E) E$ d缓冲区溢出及其攻击 http://www.sixthroom.com/ailan/f ;... 3&RootID=312&ID=312
8 I/ b! P8 n& z4 D缓冲区溢出原理分析 http://www.sixthroom.com/ailan/f ;... 3&RootID=313&ID=313
+ t- ^& @8 ^8 n# K7 ~- b2 {" }更多资料请到 www.sixthroom.com |