QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 2684|回复: 1
打印 上一主题 下一主题

[转载]COFF文件格式(2)

[复制链接]
字体大小: 正常 放大
Blackbird        

8

主题

2

听众

21

积分

升级  16.84%

该用户从未签到

新人进步奖

跳转到指定楼层
1#
发表于 2005-9-17 13:59 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
<><STRONG>重要声明:本文乃转载自其他社区,由于在下无法获得任何有关作者和出处的信息,所以不能在此登出,恳请作者原谅,并希望知情者能告知在下。本着资源的共享的精神,在下深信作者不会拒绝在下的转载行为。同时强烈BS数学中国的下载系统,它不但不能及时给予他人需要的帮助,还浪费了他人大量的时间,仅仅是为了获得无聊的点数,而且遗憾的是,那些点数并不能保证你真能获得帮助!</STRONG></P>" P  r3 h  ^9 A" |# v" v+ `) v
<>文件头<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>" ]& F$ N) C  F  J) I9 S6 V6 j+ h7 N
<>具体数值和说明请见下表:</P>8 |$ V7 G1 h  C, V* i" D/ O
<>5 a  j; J5 p2 @( H: X
<TABLE borderColor=#cccccc cellSpacing=2 cellPadding=2 width="100%" align=center bgColor=#ffffff border=1>0 W% \/ g, G( L; r, a
; i5 d  \' R- N, J
<TR>6 e9 D$ x8 O, L6 Z1 b' U
<TD>1 O7 p/ N! q0 ~( Z1 m
<>值</P></TD>
+ k2 ?( m4 ^' _) ]9 f. X3 j<TD>
8 O6 k- w, G4 [6 j; |* q8 r! I' q* S6 J<>名称</P></TD>
9 M" ]6 U) V8 G. p6 i6 d7 \<TD>
% e. L  m9 o+ J1 x% f7 h! o, }4 n<>说明</P></TD>7 t- K. a5 d" L/ H, l8 |9 D
<TR>% h+ P8 k/ H' k/ p" ]* s6 e
<TD>
  R4 v- p1 T0 Q4 [. C<>0x0001</P></TD>* j5 W  j9 G5 I# F) Z
<TD>
8 q) Z3 a3 p% `8 a<>F_RELFLG  </P></TD>" ^) {) b3 m2 d: p6 l7 x
<TD>
2 I$ ]! r, D3 i- L, L<>无重定位信息标记。这个标记指出COFF文件中没有重定位信息。通常在目标文件中这个标记们为0,在可执行文件中为1。</P></TD>4 W, m0 b* u% O! z7 G3 M
<TR>: y" |# S) l- B7 i4 k
<TD>4 U: T  c# a) d* V6 X5 _  E( b. M
<>0x0002</P></TD>7 v5 K. a6 ?+ e2 _0 W& N
<TD>. W: J, t$ a! N% ~  J+ w
<>F_EXEC</P></TD>
  c! }! q5 {' M' k: O7 V<TD>
$ z0 V( T5 ^7 q: C2 \4 ]% Z/ `<>可执行标记。这个标记指出 COFF</FONT> 文件中所有符号已经解析, COFF</FONT> 文件应该被认为是可执行文件。</P></TD>
" X* h" P/ B6 v5 H<TR>! L) ^$ W6 W5 e0 P- J5 P% ^, O* q
<TD>( V$ _: }- k3 y# e, {
<>0x0004</P></TD>! i" n) x7 V/ k8 ?: g3 r: M
<TD>& F/ v& c. n+ |* [* i
<>F_LNNO</P></TD>
0 k& D! ]$ I' @& |; y2 Z<TD>
( X3 u) z. K( ~+ i+ [/ H<>可执行标记。这个标记指出 COFF</FONT> 文件中所有符号已经解析, COFF</FONT> 文件应该被认为是可执行文件。</P></TD>
3 B8 u, i: A1 {" y<TR>; O  B* S: L7 q
<TD>' V% R( b0 k% ^  M; V# U6 E9 `0 D
<>0x0008</P></TD>
9 \, K$ W! F" t# U3 j; g<TD>
. C& z' l# h& {% J& x4 V' n<>F_LSYMS</P></TD>9 A; i7 b! J* r' B
<TD>6 p2 |6 W& Z0 P" ]" q. E
<>文件中的符号信息已经被去掉。</P></TD>
3 j$ }/ [4 L$ w% b& T) N<TR>
* e) U3 U+ q3 C- y$ @& Z, r<TD>" A) F: w0 }, R& p4 b5 L: C3 Y
<>0x0100</P></TD>0 C( _6 R; ^# M; F9 u$ @# Z9 I
<TD>$ W* ~! t1 v: i
<>F_AR32WR</P></TD>7 a4 g  U* _  {9 }0 @" Z
<TD>
2 d' l4 q* j  `; f+ W: _  {<>些标记指出文件是 32 位的 Little-Endian COFF文件。</P></TD></TR></TABLE></P>9 I0 y' n: a2 [+ K
<>注: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>
& B$ A9 x$ ]9 f
[此贴子已经被作者于2005-9-17 14:50:44编辑过]
zan
转播转播0 分享淘帖0 分享分享0 收藏收藏0 支持支持0 反对反对0 微信微信

1

主题

2

听众

60

积分

升级  57.89%

该用户从未签到

新人进步奖

回复

使用道具 举报

您需要登录后才可以回帖 登录 | 注册地址

qq
收缩
  • 电话咨询

  • 04714969085
fastpost

关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

手机版|Archiver| |繁體中文 手机客户端  

蒙公网安备 15010502000194号

Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

GMT+8, 2026-4-20 21:18 , Processed in 0.478786 second(s), 58 queries .

回顶部