- 在线时间
- 0 小时
- 最后登录
- 2005-9-17
- 注册时间
- 2005-9-17
- 听众数
- 2
- 收听数
- 0
- 能力
- 0 分
- 体力
- 55 点
- 威望
- 0 点
- 阅读权限
- 20
- 积分
- 21
- 相册
- 0
- 日志
- 0
- 记录
- 0
- 帖子
- 8
- 主题
- 8
- 精华
- 0
- 分享
- 0
- 好友
- 0
升级   16.84% 该用户从未签到
 |
< ><STRONG>重要声明:本文乃转载自其他社区,由于在下无法获得任何有关作者和出处的信息,所以不能在此登出,恳请作者原谅,并希望知情者能告知在下。本着资源的共享的精神,在下深信作者不会拒绝在下的转载行为。同时强烈BS数学中国的下载系统,它不但不能及时给予他人需要的帮助,还浪费了他人大量的时间,仅仅是为了获得无聊的点数,而且遗憾的是,那些点数并不能保证你真能获得帮助!</STRONG></P>5 A5 i; K, w, J% z
< >文件头<br> 文件头,自然是从文件的0偏移处开始,它的结构很简单。用C的结构描述如下:<br>typedef struct {<br> unsigned short usMagic; // 魔法数字<br> unsigned short usNumSec; // 段落(Section)数<br> unsigned long ulTime; // 时间戳<br> unsigned long ulSymbolOffset; // 符号表偏移<br> unsigned long ulNumSymbol; // 符号数<br> unsigned short usOptHdrSZ; // 可选头长度<br> unsigned short usFlags; // 文件标记<br>} FILEHDR;<br> 结构中usMagic成员是一个魔法数字(Magic Number),在I386平台上的COFF文件中它的值为0x014c。如果COFF文件头中魔法数字不为0x014c,那就不用看了,这不是一个I386平台的COFF文件。其实这就是一个平台标识。<br> 第二个成员usNumSec是一个无符号短整型,它用来描述段落的数量。段落头(Section Header)的数目就是它。<br> ulTime成员是一个时间戳,它用来描述COFF文件的建立时间。当COFF文件为一个可执行文件时,这个时间戳经常用来当做一个加密用的比对标识。<br> ulSymbolOffset是符号表在文件中的偏移量,这是一个绝对偏移量,要从文件头开始计数。在COFF文件的其它节中,也存在这种偏移量,它们都是绝对偏移量。<br> ulNumSymbol成员给出了符号表中符号记录的数量。<br> usOptHdrSZ是可选头的长度,通常它为0。而可选头的类型也是从这个长度得知的,针对不同的长度,我们就要选择不同的处理方式。<br> usFlag是COFF文件的属性标记,它标识了COFF文件的类型,COFF文件中所保存的数据等等信息。</P>
' X U8 D* V% H8 _* ^( @3 ` T9 }< >具体数值和说明请见下表:</P>( G* F" O, _" j8 m, v: a
< >
. ?4 _0 V7 \3 {/ f c<TABLE borderColor=#cccccc cellSpacing=2 cellPadding=2 width="100%" align=center bgColor=#ffffff border=1>* Y* M" V! c/ W
& q; ?3 A" S8 |2 @+ T<TR>
4 m" u x4 J, i6 g, H<TD>1 T2 Y W2 K3 z. o* m
< >值</P></TD>$ k D' H K( A0 S' Y4 _& r* B
<TD>* S, U; F ~( S# Y9 X
< >名称</P></TD>2 T; ]" C5 N9 Y. w4 O
<TD>9 c q L5 ^$ l$ g$ e: M
< >说明</P></TD>
6 p8 t4 z) e, ~+ w: O% g% F<TR>, m8 X) |4 s% T# R: ~0 y
<TD>
/ H+ j0 x2 a, y, S9 D< >0x0001</P></TD>2 v- R( V& u# m e* X0 h" b A2 X
<TD>( t2 E7 W' O' |( G$ O7 Q( `
< >F_RELFLG </P></TD>
; e8 i# Y: Q2 ?8 r# D T<TD>: ]+ v- Q. s; v' T7 ^1 A
< >无重定位信息标记。这个标记指出COFF文件中没有重定位信息。通常在目标文件中这个标记们为0,在可执行文件中为1。</P></TD>5 c- a# l8 S5 b% y
<TR>
& l/ G6 r1 ]& i* {<TD>* Y# ?5 z# L% E, [
< >0x0002</P></TD>
. Y- M8 `. Z2 y% D<TD>: D) d% r) a1 t! t( c
< >F_EXEC</P></TD>, P, T8 m3 |) i0 I
<TD>
: j G' l$ A) Y3 o7 V5 f) _< >可执行标记。这个标记指出 COFF</FONT> 文件中所有符号已经解析, COFF</FONT> 文件应该被认为是可执行文件。</P></TD>" }4 a" D- s$ z4 z( Y- @# u5 z8 f
<TR>
$ I9 i5 _3 o% L- b<TD>0 E. X$ N0 O$ g3 M% a' p1 t
< >0x0004</P></TD>
}5 G3 V4 F+ W7 a: {6 W+ X<TD>
* f1 t- n% ~. K7 I# L< >F_LNNO</P></TD>0 @7 ?( E' j4 A% o7 R h
<TD>
! ?7 M: O6 L6 ?. a3 P* K8 M5 r< >可执行标记。这个标记指出 COFF</FONT> 文件中所有符号已经解析, COFF</FONT> 文件应该被认为是可执行文件。</P></TD>
7 W2 ]5 X( i8 x2 A<TR>
/ o @8 l+ O/ @, o4 r; Y+ @! v<TD>; u, m, A: e, y* n
< >0x0008</P></TD>
* \7 H8 H! V! ]! R! C1 n* n ^: e<TD>5 y* w: m0 z' ^
< >F_LSYMS</P></TD>2 P& {; D7 J) H B3 T/ c j
<TD>
' A5 Z2 M. E6 U7 z, l6 L< >文件中的符号信息已经被去掉。</P></TD># Y4 A, F1 p5 D- {9 y# f" M
<TR>
9 t/ k L1 E- ^4 w2 K<TD>
4 G2 i {6 ]7 c2 X( ?+ @/ `< >0x0100</P></TD>
8 j$ {/ J9 [; t+ ]9 n<TD>, K" w* L, m1 Y' ?6 x
< >F_AR32WR</P></TD>' |) c' c- i+ G" R# S6 L6 P# h7 R8 \
<TD>3 d+ X, g7 A( {- |- y
< >些标记指出文件是 32 位的 Little-Endian COFF文件。</P></TD></TR></TABLE></P>% `' H# U7 @/ ]% s" x( }1 A% }. Y- Y
< >注:Little-Endian,记不得它的中文名称了。它是指数据的排列方式。比如:十六进制的0x1234以Little-Endian方式在内存中的顺序为0x34 0x12。与之相反的是Big-Endian,这种方式下,在内存中的顺序是0x12 0x34。<br>这个表的内容并不全面,但在目标文件中,常用的也就只有这些。其它的标记我将在以后介绍PE格式时给出。<br>可选头<br> 可选头接在文件头的后面,也就是从COFF文件的0x0014偏移处开始。长度可以为0。不同长度的可选头,其结构也不同。标准的可选头长度为24或28字节,通常是28啦。这里我就只介绍长度为28的可选头。(因为这种头的长度是自定义的,不同的人定义的结果就不一样,我只能选一种最常用的头来介绍,别的我也不知道)<br>这种头的结构如下:<br>typedef struct {<br> unsigned short usMagic; // 魔法数字<br> unsigned short usVersion; // 版本标识<br> unsigned long ulTextSize; // 正文(text)段大小<br> unsigned long ulInitDataSZ; // 已初始化数据段大小<br> unsigned long ulUninitDataSZ; // 未初始化数据段大小<br> unsigned long ulEntry; // 入口点<br> unsigned long ulTextBase; // 正文段基址<br> unsigned long ulDataBase; // 数据段基址(在PE32中才有)<br>} OPTHDR;<br> 第一个成员usMagic还是魔法数字,不过这回它的值应该为0x010b或0x0107。当值为0x010b时,说明COFF文件是一个一般的可执行文件;当值为,0x0107时,COFF则为一个ROM镜像文件。<br> usVersion是COFF文件的版本,ulTextSize是这个可执行COFF的正文段长度,ulInitDataSZ和ulUninitDataSZ分别为已初始化数据段和未初始化数据段的长度。<br> ulEntry是程序的入口点,也就是COFF载入内存时正文段的位置(EIP寄存器的值),当COFF文件是一个动态库时,入口点也就是动态库的入口函数。<br> ulTextBase是正文段的基址。<br> ulDataBase是数据段基址。<br> 其实在这些成员中,只要注意usMagic和ulEntry就可以了。</P>
1 v, E) o, W+ E Y0 F[此贴子已经被作者于2005-9-17 14:50:44编辑过] |
zan
|