|
EDA365欢迎您登录!
您需要 登录 才可以下载或查看,没有帐号?注册
x
缓冲区溢出的危害及避免缓冲区溢出的三种方法2 S: j1 V/ r/ |6 T6 I/ @! Y; e! h
面试官不讲武德,居然让我讲讲蠕虫和金丝雀!
; l* u# m# ^& L9 c$ \% q1. 蠕虫病毒简介4 \4 G$ ~" f3 T$ V) H9 P$ s) P
2. 缓冲区溢出& _( N+ N8 }8 L5 M6 s6 c
3. 缓冲区溢出举例/ D) r/ @8 p# N- c: s: P. ]9 A
4. 缓冲区溢出的危害* N) ^7 a- [) c1 ~: B, X K: K
5. 内存在计算机中的排布方式
6 g9 l' n5 f1 D6. 计算机中越界访问的后果6 ^$ D& j- j+ @- \
7. 避免缓冲区溢出的三种方法
5 p$ c, L$ V# c5 h9 \7.1 栈随机化
6 M8 }3 S7 B4 B/ [7.2 检测栈是否被破坏
# J6 J( s! v$ { a7.3 限制可执行代码区域; W$ ^5 H6 ~) f Z/ |1 X6 ~
8. 总结 ~* H3 ]6 n, t6 V( a* \6 d s, d2 j
) F$ K/ Y) V: @蠕虫病毒是一种常见的利用Unix系统中的缺点来进行攻击的病毒。缓冲区溢出一个常见的后果是:黑客利用函数调# o% ^9 i8 C5 L; b2 V( Y
用过程中程序的返回地址,将存放这块地址的指针精准指向计算机中存放攻击代码的位置,造成程序异常中止。为: G) C6 v- t, _0 o7 @
了防止发生严重的后果,计算机会采用栈随机化,利用金丝雀值检查破坏栈,限制代码可执行区域等方法来尽量避
1 c- M7 i1 G4 h免被攻击。虽然,现代计算机已经可以“智能”查错了,但是我们还是要养成良好的编程习惯,尽量避免写出有漏洞
: `& h2 w! }- F+ a的代码,以节省宝贵的时间!
9 c4 N5 e& b+ e- S8 i1. 蠕虫病毒简介蠕虫病毒简介. U( h+ U; T5 l; m/ J
蠕虫是一种可以自我复制的代码,并且通过网络传播,通常无需人为干预就能传播。蠕虫病毒入侵并完全控制一台计算机# i" M5 }/ {) s
之后,就会把这台机器作为宿主,进而扫描并感染其他计算机。当这些新的被蠕虫入侵的计算机被控制之后,蠕虫会以这些计
& e1 e9 i! r2 D% U) L& X算机为宿主继续扫描并感染其他计算机,这种行为会一直延续下去。蠕虫使用这种递归的方法递归的方法进行传播,按照指数增长指数增长的规
& g1 i* ~5 a7 c7 u* X, U律分布自己,进而及时控制越来越多的计算机。7 M1 J+ P% l1 R" F" S
2. 缓冲区溢出缓冲区溢出* t$ ^, N6 u6 q
缓冲区溢出是指计算机向缓冲区内填充数据位数时超过了缓冲区本身的容量,溢出的数据覆盖在合法数据上计算机向缓冲区内填充数据位数时超过了缓冲区本身的容量,溢出的数据覆盖在合法数据上。理想的情
, V$ g/ _, {5 f* W% f6 g况是:程序会检查数据长度,而且并不允许输入超过缓冲区长度的字符。但是绝大多数程序都会假设数据长度总是与所分绝大多数程序都会假设数据长度总是与所分' u8 `' i/ h7 d Y/ m" A3 v
配的储存空间相匹配配的储存空间相匹配,这就为缓冲区溢出埋下隐患。操作系统所使用的缓冲区,又被称为“堆栈”,在各个操作进程之间,指9 l% ]6 ?% K2 _# O- R
令会被临时储存在“堆栈”当中,“堆栈堆栈”也会出现缓冲区溢出也会出现缓冲区溢出。
( ^3 F' h6 \' a2 @' }9 ~' U* G: Z3. 缓冲区溢出举例缓冲区溢出举例; m. z: D4 ?5 b1 M' q
void echo()
/ a# @+ h3 q9 R% O' i8 i8 _% K* o{ char buf[4]; /*buf故意设置很小*/- u6 E1 a" v8 S/ i: Z
gets(buf);
) Q" `' v j; h" K h, j' ^% c lputs(buf);
% P9 `& Z' q* d+ {}8 M% Z+ Q2 T/ T: J- X8 Z9 ^
void call_echo()
3 S- W, c, a+ B6 U; F9 @* o- b{ echo();
/ J$ I1 S) s0 X9 k}1 g. W1 E( c# C8 x. r. B+ e
反汇编如下:1 Q$ `* `1 ?. \ {/ F
/*echo*/
4 S0 Y- Q. c* E# f( o000000000040069c <echo>:: I, O; @! Q; t/ S, _: p9 G
40069c:48 83 ec 18 sub $0x18,%rsp /*0X18 == 24,分配了24字节内存。计算机会多分配一些给缓冲区*/2 ]; S+ Q3 W. b V
14006a0:48 89 e7 mov %rsp,%rdi: Z2 L3 a; G7 X* C# @
4006a3:e8 a5 ff ff ff callq 40064d <gets>
2 \, L! a3 E$ f8 z4 v4 |- A6 k; `4006a8::48 89 e7 mov %rsp,%rdi
) w) O8 j& h+ v# K& e" X4006ab:e8 50 fe ff ff callq callq 400500 <puts@plt>8 i6 G, k, ]0 y8 m6 X
4006b0:48 83 c4 18 add $0x18,%rsp
; Z7 I% h) X+ Z& T6 z* \. L- V4006b4:c3 retq
; d1 z' O0 w5 \# h: f8 U/*call_echo*/
. n2 L. E+ I$ Z8 Q1 ^4006b5:48 83 ec 08 sub $0x8,%rsp. O1 G5 j( i+ x- x+ h8 s
4006b9:b8 00 00 00 00 mov $0x0,%eax
- m, J/ E0 T& T9 P i1 K4006be:e8 d9 ff ff ff callq 40069c <echo>) W. L; ~ l& \( T5 C. R
4006c3:48 83 c4 08 add $0x8,%rsp: d$ K U1 U5 e/ {1 W, i
4006c7:c3 retq8 ^4 K8 q v& n8 Q/ g% c( D
在这个例子中,我们故意把buf设置的很小。运行该程序,我们在命令行中输入012345678901234567890123,程序立马就会报5 E# C% c0 r% r
错:Segmentation fault。, ^ f) t( O. ^; G# o( _# w6 t4 z, n
要想明白为什么会报错,我们需要通过分析反汇编来了解其在内存是如何分布的在内存是如何分布的。具体如下图所示:
+ c9 b, n' [6 d- b4 y8 x9 ~如下图所示,此时计算机为buf分配了24字节空间,其中20字节还未使用。
9 X- p0 {5 }! e i此时,准备调用echo函数,将其返回地址压栈。
- o' n6 q7 z) |+ F1 u, ^+ a当我们输入“0123456789012345678 9012"时,缓冲区已经溢出,但是并没有破坏程序的运行状态并没有破坏程序的运行状态。" C% |% V) c, R# `0 |3 c
当我们输入:“012345678901234567 890123"。缓冲区溢出,返回地址被破坏返回地址被破坏,程序返回 0x0400600。: A1 j4 v. R) e8 p# Z+ o
这样程序就跳转到了计算机中其他内存的位置,很大可能这块内存已经被使用。跳转修改了原来的值跳转修改了原来的值,所以程序就会中止运
% C4 @9 O' b0 S( ~行。
7 q' v$ n' p, s' t) p8 V黑客可以利用这个漏洞,将程序精准跳转到其存放木马的位置(如nop sled技术),然后就会执行木马程序,对我们的计算机6 t! m% a' T* E. W
造成破坏。5 t% U1 j6 l+ H/ k4 y
4. 缓冲区溢出的危害缓冲区溢出的危害
( m, X/ ]8 W, d1 |缓冲区溢出可以执行非授权指令,甚至可以取得系统特权,进而进行各种非法操作。第一个缓冲区溢出攻击--Morris蠕虫,发
( B3 g1 Q, K$ v生在二十年前,它曾造成了全世界6000多台网络服务器瘫痪。2 s; {4 {& q9 ]' N- R! d
在当前网络与分布式系统安全中,被广泛利用的50%以上都是缓冲区溢出,其中最著名的例子是1988年利用fingerd漏洞的蠕
' c! Y- m5 }! C1 r, T虫。而缓冲区溢出中,最为危险的是堆栈溢出堆栈溢出。因为入侵者可以利用堆栈溢出,在函数返回时改变返回程序的地址,让其函数返回时改变返回程序的地址,让其- v+ c9 q( D: y+ R9 |
跳转到任意地址跳转到任意地址。带来的危害有两种,一种是程序崩溃导致拒绝服务,另外一种就是跳转并且执行一段恶意代码,比如得到
* }) D5 s/ c; M# d' l7 Lshell,然后为所欲为。3 X9 s+ H( M+ d0 Z4 \. E) Y
5. 内存在计算机中的排布方式内存在计算机中的排布方式
0 e: [. x# {+ b: r; [4 w3 }- a内存在计算机中的排布方式如下,从上到下依次为共享库,栈,堆,数据段,代码段。各个段的作用简介如下:7 g, H( d+ u( t$ }8 v. h
共享库共享库:共享库以.so结尾.(so==share object)在程序的链接时候并不像静态库那样在拷贝使用函数的代码,而只是作些标记。然后
. n. e2 q2 }1 m, X+ |在程序开始启动运行的时候,动态地加载所需模块。所以,应用程序在运行的时候仍然需要共享库的支持。共享库链接出来的1 A( O" ?9 M; M( @
文件比静态库要小得多。
9 N8 e) \7 x; R& b栈栈:栈又称堆栈,是用户存放程序临时创建的变量程序临时创建的变量,也就是我们函数{}中定义的变量,但不包括但不包括static声明的变量声明的变量,static意
% Y+ c3 ?* A2 j味着在数据段中存放变量数据段中存放变量。
% g$ I. z4 k' y除此之外,在函数被调用时,其参数也会被压入发起调用的进程栈中参数也会被压入发起调用的进程栈中,并且待到调用结束后,函数的返回值也会被存放回的返回值也会被存放回
) M4 v! U9 ~7 p2 a5 U. \栈中栈中,由于栈的先进后出特点,所以栈特别方便用来保存、恢复调用现场。从这个意义上讲,我们可以把堆栈看成一个寄
! q5 J- m9 o+ j- E7 l1 `" }存,交换临时数据的内存区。在X86-64 Linux系统中,栈的大小一般为8M(用ulitmit - a命令可以查看)。; U2 ?; G1 K, |+ A0 n! R2 Q( c7 @
堆堆:堆是用来存放进程中被动态分配的内存段,它的大小并不固定,可动态扩张或缩减大小并不固定,可动态扩张或缩减。当进程调用malloc等函数分配内存0 q) Q6 \: J1 K/ Q
时,新分配的内存就被动态分配到堆上,当利用free等函数释放内存时,被释放的内存从堆中被剔除。7 C9 L |2 i0 ]! ?3 `1 C2 @, k* y) `# W
堆存放new出来的对象,栈里面所有对象都是在堆里面有指向的。假如栈里指向堆的指针被删除,堆里的对象也要释放
3 e3 ] e9 D5 T. H$ e0 s& @(C++需要手动释放)。当然现在面向对象程序都有'垃圾回收机制',会定期的把堆里没用的对象清除出去。6 C: h& M0 N/ \* {* d" \0 B* o
2数据段数据段:数据段通常用来存放程序中已初始化的全局变量和已初始化为非已初始化的全局变量和已初始化为非0的静态变量的静态变量的一块内存区域,属于静态内存分
G/ T5 { q# h9 z9 |6 O配。直观理解就是C语言程序中的全局变量(注意:全局变量才算是程序的数据,局部变量不算程序的数据,只能算是函全局变量才算是程序的数据,局部变量不算程序的数据,只能算是函
: W4 }1 @, i! u数的数据数的数据)7 l+ ~2 O L: i- Z6 ?6 y
代码段代码段:代码段通常用来存放程序执行代码执行代码的一块区域。这部分区域的大小在程序运行前就已经确定了,通常这块内存区域$ Z0 z' I' G' d, [
属于只读只读,有些架构也允许可写,在代码段中也有可能包含以下只读的常数变量,例如字符串常量等。
4 ~" h3 l' G( z8 l0 q下面举个例子来看下代码中各个部分在计算机中是如何排布的。- E. A9 v6 J; O% s- }$ }
#include <stdio.h>
- C- N# y0 {8 e+ J: T- Y% M#include <stdlib.h>
# S& c& i( Y+ q- x, v! E' `# dchar big_array[1L<<24]; /*16 MB*/
; D8 R# D4 Y& Jchar huge_array[1L<<31]; /*2 GB*/$ |+ G" Y+ F7 \' D* V
int global = 0;
5 G! V$ l( e4 Y* m8 Q9 iint useless() {return 0;}. C2 P) t! T2 B0 d5 J; W$ [) q
int main()
$ {3 o8 I4 C& k# i* |* n) S' [{ void *phuge1,*psmall2,*phuge3,*psmall4;
& H, K9 z2 p( V0 H; h: B tint local = 0;
5 w' z% o( {, V7 g- n% qphuge1 = malloc(1L<<28); /*256 MB*/
- _6 f5 {7 ?5 B8 [. h" Lpsmall2 = malloc(1L<<8); /*256 B*/$ n+ p i) \$ C2 r
phuge3 = malloc(1L<<32); /*4 GB*/* b3 l" s2 s4 T8 R
psmall4 = malloc(1L<<8); /*256 B*/. A3 o+ a6 e; M! u2 k
}
' ~ z c' m3 P上述代码中,程序中的各个变量在内存的排布方式如下图所示。根据颜色可以一一对应起来。由于了local变量存放在栈区,四
l0 t5 N. K1 e. \" ]个指针变量使用了malloc分配了空间, 所以存放在堆上,两个数组big_ array,huge_array存放在数据段,main,useless函数的其1 Z+ G; S( H" O6 ~9 J3 N& R
他部分存放在代码段中。+ m& R$ f. ?! M) _" c' n3 x
6. 计算机中越界访问的后果计算机中越界访问的后果+ F! o8 f) p3 j$ D& M' i. x! V) H
下面再看一个例子,看下越界访问内存会有什么结果。% x/ n3 t e. j
typedef struct
7 O' m- Z6 c: k( c* ^{ int a[2];
7 y: g% X( _' gdouble d;/ |3 ^. l/ T9 ?& B- _
}struct_t;
" a5 F2 Y( [3 W2 B2 Odouble fun(int i)/ B( Q! [, e/ n+ R& {0 O7 _: S
{! f2 A( t3 o- q* C8 Y+ _6 y3 p' _
volatile struct_t s;" z& w7 c; \9 N5 l
s.d = 3.14;
. {: d8 u c$ Ds.a[i] = 1073741824; /*可能越界*/
0 [$ Q* N4 K: p: [& vreturn s.d;
/ m) p d3 v# H! I* j4 Z}
! |' p' x- @) M1 x- n; Pint main()
4 ]! W% T+ P) Y1 h7 j( y" X{9 j' s2 I, {) ^% i8 k' ^
printf("fun(0):%lf\n",fun(0));2 m8 n9 G1 K0 g; ]
printf("fun(1):%lf\n",fun(1));
$ _- ?( H' b8 E5 Pprintf("fun(2):%lf\n",fun(2));9 r5 v0 ^. d% ?! N# L
printf("fun(3):%lf\n",fun(3));
9 R4 ^) ?3 q3 y8 w+ \" f3 O$ _printf("fun(6):%lf\n",fun(6));
; C5 g/ }( {& t( C _return 0;
8 A5 V4 z" w3 Y1 N3 t& z) h}
6 ?% \7 [2 [5 U( A打印结果如下所示:. D1 }2 x5 s: Y) l8 E
fun(0):3.14
- k; U4 l; n5 hfun(1):3.14: ^4 d/ c2 ?6 E% b. C& k9 w
fun(2):3.1399998664856
' O3 v9 Y- N' f1 \/ e& Q& efun(3):2.00000061035156
' {, C1 ~( m* |, t$ r3 gfun(6):Segmentation fault
" m+ M' t0 n2 I8 J( A) v在上面的程序中,我们定义了一个结构体,其中 a 数组中包含两个整数值,还有 d 一个双精度浮点数。在函数fun中,fun函数
( r+ x& |* Z$ M9 L. y" S根据传入的参数i来初始化a数组。显然,i的值只能为的值只能为0和和1。在fun函数中,同时还设置了d的值为3.14。当我们给fun函数传入0- i4 a* w# m1 E7 u X
和1时可以打印出正确的结果3.14。但是当我们传入2,3,6时,奇怪的现象发生了。为什么为什么fun((2)和)和fun((3)的值会接近)的值会接近1 x7 A$ d3 i) E7 s9 u, F
3.14,而,而fun((6)会报错呢?)会报错呢?( m3 o3 X. q5 u3 \5 H* b
3要搞清楚这个问题,我们要明白结构体在内存中是如何存储的,具体如下图所示。
0 |" P; o+ l P0 }8 F5 T- V) Q; v! o结构体在内存中的存储方式+ Q6 w/ S: l" Z% S7 A- a) U3 a" H
GCC默认不检查数组越界(除非加编译选项)。而越界会修改某些内存的值而越界会修改某些内存的值,得出我们意想不到的结果。即使有些数据相隔
2 I8 ~& w3 A# i5 T0 l% u' X万里,也可能受到影响。当一个系统这几天运行正常时,过几天可能就会崩溃。(如果这个系统是运行在我们的心脏起搏器,( ?4 F; f7 R' K; X9 A% T
又或者是航天飞行器上,那么这无疑将会造成巨大的损失!)# r0 O+ z/ o; K* f7 G. m
如上图所示,对于最下面的两个元素,每个块代表 4 字节。a数组占用8个字节,d变量占用8字节,d排布在a数组的上方。所以6 @9 j6 D* J' ~$ d% d8 C% p+ v0 t
我们会看到,如果我引用 a[0] 或者 a[1],会按照正常修改该数组的值。但是当我调用 fun(2) 或者 fun(3)时,实际上修改的是这实际上修改的是这$ t; R' L8 g, O
个浮点数个浮点数 d 所对应的内存位置所对应的内存位置。这就是为什么我们打印出来的fun(2)和fun(3)的值如此接近3.14的原因。8 a, k! V" \9 ?& _
当输入 6 时,就修改了对应的这块内存的值。原来这块内存可能存储了其他用于维持程序运行的内容,而且是已经分配原来这块内存可能存储了其他用于维持程序运行的内容,而且是已经分配, F9 W& Z' B) ^' n0 r( l9 S: B" S
的内存的内存。所以,我们程序就会报出Segmentation fault的错误。
( }# N& E) P; s8 N7. 避免缓冲区溢出的三种方法避免缓冲区溢出的三种方法/ ^* J8 j e& ]5 h" L
为了在系统中插入攻击代码,攻击者既要插入代码插入代码,也要插入指向这段代码的指针这段代码的指针。这个指针也是攻击字符串的一部分。产" n+ p6 _& I. z: o b7 T
生这个指针需要知道这个字符串放置的栈地址栈地址。在过去,程序的栈地址非常容易预测。对于所有运行同样程序和操作系统版7 g1 i9 w( C0 r0 J* F- ~
本的系统来说,在不同的机器之间,栈的位置是相当固定的栈的位置是相当固定的。因此,如果攻击者可以确定一个常见的Web服务器所使用的栈+ ~- R1 ]0 z) |$ r6 O! r
空间,就可以设计一个在许多机器上都能实施的攻击。, b8 W* q1 y; f
7.1 栈随机化栈随机化( _; O2 `' D ?8 Z
栈随机化的思想使得栈的位置在程序每次运行时都有变化栈的位置在程序每次运行时都有变化。因此,即使许多机器都运行同样的代码,它们的栈地址都是不同
/ T9 B6 A. i/ ?的。实现的方式是:程序开始时,在栈上分配一段0 ~ n字节之间的随机大小的空间,例如,使用分配函数alloca在栈上分配指0 F/ n( M/ M2 d! p; h
定字节数量的空间。程序不使用这段空间程序不使用这段空间,但是它会导致程序每次执行时后续的栈位置发生了变化程序每次执行时后续的栈位置发生了变化。分配的范围n必须足0 }0 \% _" E5 A% y
够大,才能获得足够多的栈地址变化,但是又要足够小,不至于浪费程序太多的空间。
6 ?: T& v: l) Iint main()
; h. w% Y3 J. B, m; n( g5 o{long local;
4 w6 R% I* L8 u, m1 V' ~. H4 U" u1 fprintf("local at %p\n",&local);
* R, r. M( t; y r. c- ]5 Ireturn 0;- l1 {! H# S7 n4 o
}4 w+ Z# |4 y' M4 m' q
这段代码只是简单地打印出main函数中局部变量的地址。在32位 Linux上运行这段代码10000次,这个地址的变化范围为
r& g. Y- [" o; J) A8 C0 o0xff7fc59c到0xffffd09c,范围大小大约是 。在64位 Linux机器上运行,这个地址的变化范围为0x7fff0001b698到0x7ffffffaa4a8,范
0 w$ w/ S" k: W& A% S. ~围大小大约是 。5 \0 u) p6 f* w3 U; m& R
其实,一个好的黑客专家,可以使用暴力破坏栈的随机化。对于32位的机器,我们枚举 个地址就能猜出来栈的地址。7 x& T0 T/ D A" M* q% U7 {/ ?
对于64位的机器,我们需要枚举 次。如此看来,栈的随机化降低了病毒或者蠕虫的传播速度,但是也不能提供完全的
$ s! O6 X) ^ n( M/ ]# L安全保障。! i, E8 D5 o' z+ R6 r
7.2 检测栈是否被破坏检测栈是否被破坏0 b4 p! o9 [7 U S, j- a4 n
计算机的第二道防线是能够检测到何时栈已经被破坏。我们在echo函数示例中看到,当访问缓冲区越界时,会破坏程序的运行: J! |+ z2 \0 G2 O) y/ w
状态。在C语言中,没有可靠的方法来防止对数组的越界写。但是,我们能够在发生了越界写的时候,在造成任何有害结果之
$ u. c# o" |$ L1 e0 r前,尝试检测到它。. r+ B, A. r% o9 U0 Q8 Z `5 M6 b" T
GCC在产生的代码中加人了一种栈保护者机制,来检测缓冲区越界。其思想是在栈帧中任何局部缓冲区与栈状态之间存储在栈帧中任何局部缓冲区与栈状态之间存储
: Z# @* u! e- P! p( d一个特殊的金丝雀值一个特殊的金丝雀值,如下图所示:% j; |+ d3 V1 z2 t9 _1 v' S7 t4 p
这个金丝雀值,也称为哨兵值,是在程序每次运行时随机产生的,因此,攻击者很难猜出这个哨兵值。在恢复寄存器状态和从
; {5 ~( K6 A1 p* q9 T8 [+ x) Z2 U/ r* O函数返回之前,程序检查这个金丝雀值是否被该函数的某个操作或者该函数调用的某个函数的某个操作改变了程序检查这个金丝雀值是否被该函数的某个操作或者该函数调用的某个函数的某个操作改变了。如果是7 d7 p4 d% b1 g$ E, O
的,那么程序异常中止。8 c1 V! V" [: C& S$ x& X6 w9 K
英国矿井饲养金丝雀的历史大约起始1911年。当时,矿井工作条件差,矿工在下井时时常冒着中毒的生命危险。后
& Y# O. d0 v! f2 z7 h( d7 q来,约翰·斯科特·霍尔丹(John Scott Haldane)在经过对一氧化碳一番研究之后,开始推荐在煤矿中使用金丝雀检测
e6 c# w2 H$ P6 ^, P一氧化碳和其他有毒气体。金丝雀的特点是极易受有毒气体的侵害,因为它们平常飞行高度很高,需要吸入大量空
* H6 u/ R1 B8 E( r9 U8 ?3 Z气吸入足够氧气。因此,相比于老鼠或其他容易携带的动物,金丝雀会吸入更多的空气以及空气中可能含有的有毒
- y6 \: ^& y0 W3 d* a8 A3 T( {物质。这样,一旦金丝雀出了事,矿工就会迅速意识到矿井中的有毒气体浓度过高,他们已经陷入危险之中,从而
; v9 g( t+ I+ P- s6 n' f及时撤离。/ P: M6 p$ Y4 _1 I! O5 x% J
4GCC会试着确定一个函数是否容易遭受栈溢出攻击,并且自动插入这种溢出检测。实际上,对于前面的栈溢出展示,我们可
1 E/ T Q0 Q1 s6 G- x4 U以使用命令行选项“-fno- stack- protector”来阻止GCC产生这种代码。当用这个选项来编译echo函数时(允许使用栈保护),得到
P4 m- z4 U; Z/ h' O下面的汇编代码
" l( ^; X+ ]) J# f//void echo
0 V# a; R0 S2 `7 M# V" bsubq $24,%rsp Allocate 24 bytes on stack" s# I* m9 y" P M6 j" F
movq %fs:40,%rax Retrieve canary9 E6 e0 q! o* {* c
movq %rax,8(%rsp) Store on stack* t7 X j7 j* J& H" l# g3 {
xorl %eax, %eax Zero out register //从内存中读出一个值9 x* l2 p, @( d& z! B. q: v9 d. [
movq %rsp, %rdi Compute buf as %rsp x9 m% j% t5 T {8 v9 a
call gets Call gets7 J6 K; Z4 L \0 f! Y o( y# K7 @. o
movq ‰rsp,%rdi Compute buf as %rsp
7 O3 [9 J" p+ R. u# t! k1 Icall puts Call puts6 F* \6 ]2 j4 O8 y# k- f
movq 8(%rsp),%rax Retrieve canary
2 [! J" H( G# _7 b) N0 Lxorq %fs:40,%rax Compare to stored value //函数将存储在栈位置处的值与金丝雀值做比较
0 L5 d& }- c" q, u) rje .L9 If =, goto ok# S0 R; }8 E) E6 m: `4 ~3 b
call __stack_chk_fail Stack corrupted
J P4 \, P0 l- {.L9$ ?' B- k+ c; p4 Z4 m
addq $24,%rsp Deallocate stack space4 z4 {4 P/ ?" W" G2 r
ret
A( x0 j8 y% e# k. p这个版本的函数从内存中读出一个值(第4行),再把它存放在栈中相对于%rsp偏移量为8的地方。指令参数各fs:40指明金丝雀
; M* J+ n. p3 u; |, a值是用段寻址从内存中读入的。段寻址机制可以追溯到80286的寻址,而在现代系统上运行的程序中已经很少见到了。将金丝) z) X1 W1 o) u5 O1 t
雀值存放在一个特殊的段中,标记为只读只读,这样攻击者就不能覆盖存储金丝雀值这样攻击者就不能覆盖存储金丝雀值。在恢复寄存器状态和返回前,函数将存函数将存
! b7 Z# v8 l3 Y$ G. j储在栈位置处的值与金丝雀值做比较储在栈位置处的值与金丝雀值做比较(通过第10行的xorq指令)。如果两个数相同,xorq指令就会得到0,函数会按照正常的5 Z7 o7 h9 l+ m& W" m. J) p$ r% T
方式完成。非零的值表明栈上的金丝雀值被修改过,那么代码就会调用一个错误处理例程。* Z4 h5 u; {3 a
栈保护很好地防止了缓冲区溢出攻击破坏存储在程序栈上的状态。一般只会带来很小的性能损失。
2 E' W' J/ c( ~$ M* H7.3 限制可执行代码区域限制可执行代码区域1 b% b$ y5 a$ ? K' d' N- i$ h& q
最后一招是消除攻击者向系统中插入可执行代码的能力。一种方法是限制哪些内存区域能够存放可执行代码限制哪些内存区域能够存放可执行代码。在典型的程
- x1 l2 n5 x. ?7 ]6 i3 d! L& S序中,只有保存编译器产生的代码的那部分内存才需要是可执行的只有保存编译器产生的代码的那部分内存才需要是可执行的。其他部分可以被限制为只允许读和写。
$ ~% g; K! ]# H( s) L% w许多系统都有三种访问形式:读(从内存读数据)、写(存储数据到内存)和执行(将内存的内容看作机器级代码)读(从内存读数据)、写(存储数据到内存)和执行(将内存的内容看作机器级代码)。
/ ^. d( L2 F2 E3 V, p3 H0 o, b' D6 I以前,x86体系结构将读和执行访问控制合并成一个1位的标志,这样任何被标记为可读的页也都是可执行的。栈必须是既可读
. `. s: v' g* n3 w6 M/ c又可写的,因而栈上的字节也都是可执行的。已经实现的很多机制,能够限制一些页是可读但是不可执行的,然而这些机制通
7 m7 W- R4 }" |/ ?* m. F常会带来严重的性能损失。
. l/ i! C) F0 P8. 总结总结6 S. X1 T, o5 J5 T9 |( f" Z
计算机提供了多种方式来弥补我们犯错可能产生的严重后果,但是最关键的还是我们尽量减少犯错。
# B4 y$ T0 g }: r* k/ K& d例如,对于gets,strcpy等函数我们应替换为 fgets,strncpy等。在数组中,我们可以将数组的索引声明为size_t类型,从根本上防8 Z6 ~: X; P" h+ }7 F7 h
止它传递负数。此外,还可以在访问数组前来加上num小于ARRAY_MAX 语句来检查数组的上界。总之,要养成良好的编程习8 w$ G2 ], A# D; C+ Q2 F
惯,这样可以节省很多宝贵的时间。同时最后也推荐两本相关书籍如下所示。/ G: C; w+ u5 x: x& [7 G4 a/ u6 c) K
代码大全(第二版)
5 a- g& H+ o& f" }8 _1 Z% |9 m+ r高质量程序设计指南7 N2 S$ `9 j1 |1 _! K
* W6 F' N/ ~# Y5 L8 V5 @ |
|