<STRONG>
& n/ g, g7 x. R- }< ><STRONG>重要声明:本文乃转载自其他社区,由于在下无法获得任何有关作者和出处的信息,所以不能在此登出,恳请作者原谅,并希望知情者能告知在下。本着资源的共享的精神,在下深信作者不会拒绝在下的转载行为。同时强烈BS数学中国的下载系统,它不但不能及时给予他人需要的帮助,还浪费了他人大量的时间,仅仅是为了获得无聊的点数,而且遗憾的是,那些点数并不能保证你真能获得帮助!</STRONG></P>
0 Q) ]+ N% @% g/ v# @) x9 G! S< ></STRONG><STRONG>符号表<br></STRONG> 符号表是对象文件中用来保存符号信息的一张表,也是COFF文件中最为复杂的一张表。所有段落使用到的符号都在这个表里。它也是由很多条记录组成,每条记录都以如下结构保存:<br>typedef struct {<br> union {<br> char cName[8]; // 符号名称<br> struct {<br> unsigned long ulZero; // 字符串表标识<br> unsigned long ulOffset; // 字符串偏移<br> } e;<br> } e;<br> unsigned long ulValue; // 符号值<br> short iSection; // 符号所在段<br> unsigned short usType; // 符号类型<br> unsigned char usClass; // 符号存储类型<br> unsigned char usNumAux; // 符号附加记录数<br>} SYMENT;<br> cName符号名称,和前面所有的名称一样,它也是8个字节,但不同的是它在一个联合体中。和它占相同的存储空间的还有ulZero和ulOffset这两个成员。如果符号的名称只有8个字符,那很好,可以直接放到这个cName中;可是,如果名称的长度大于8个字节,这里就放不下了,只好放到字符串表中。这时候,ulZero的值就会为0,而在ulOffset中会给出我们所用的符号的名称在字符串表中的偏移。<br> 一个符号有了名称不够,它还要有值!ulValue就是这个符号所代表的值。<br> iSection成员指出了这个符号所在的段落。如果它的值为0,那么这个符号就是一个外部符号,要从其它的COFF文件中解析(连接多个目标文件就是要解析这种符号)。当它的值为-1时,说明这个符号的值是一个常量,不是它在段落中的偏移。而当它的值为-2时,这个符号只是一个调试符号,只有在调试时才会用到它。当它大于0时,才是符号所在段的索引值。<br> usType是符号的类型标识。它用来说明这个符号的类型,是函数?整型?还是其它什么。这个标识是两个字节。<br> 低字节的低四位是基本标识,它指出了符号的基本类型,如整型,字符,结构,联合等。高四位指出了符号的高级类型,如指针(0001b),函数(0010b),数组(0011b),无类型(0000b)等。现在的编译器,通常不使用基本类型,只使用高级类型。所以,符号的基本类型通常被设为0。<br>高字节通常未用。<br> usClass是符号的存储类型标识。它指明了符号的存储方式。<br> 其值与意义见下表:<br></P>3 J5 H9 j6 ?$ i$ o a" d8 v
<TABLE cellSpacing=0 borderColorDark=#ffffff cellPadding=0 width="80%" align=center bgColor=#ffffff borderColorLight=#000000 border=1> [1 ]3 x7 r0 N
?$ V4 f; c! N1 k<TR># _9 e$ v- z" t! q; \% C% ?( D
<TD vAlign=top width=55>值</TD>
5 t1 p7 a x, l8 v; w3 ~0 a; `- W<TD vAlign=top width=72>名称 </TD>+ H; K9 _7 K1 F" J% l* j- ~
<TD vAlign=top width=441>说明</TD></TR>/ r. T1 {/ O H! @9 `
<TR>
1 t) a9 C) ?+ B; a( s0 S<TD vAlign=top width=55>NULL</TD>2 ~+ U2 ? r g' {+ k( y6 Q _2 G( o
<TD vAlign=top width=72>0</TD>5 c. H( @7 p9 v+ u) y' M0 o4 h
<TD vAlign=top width=441>无存储类型。</TD></TR>) `7 |, r- q$ q7 F" N: i( v( R# j
<TR>
" b w/ i3 ]( L<TD vAlign=top width=55>AUTOMATIC</TD>
' G& D( |& J8 p% i" ?<TD vAlign=top width=72>1</TD>
6 f5 z# ~$ Y: m+ E" ~$ H<TD vAlign=top width=441>自动类型。通常是在栈中分配的变量。</TD></TR>3 n+ J; W% E- W& b. Q- @
<TR>
7 l1 c* ^, e6 Y) i7 q, i: U<TD vAlign=top width=55>EXTERNAL</TD>
4 N m6 Y1 W4 ?7 _<TD vAlign=top width=72>2</TD>% j8 }' n* ~! ^) \1 g1 P# y" {' i* K
<TD vAlign=top width=441>外部符号。当为外部符号时,iSection的值应该为0,如果不为0,则ulValue为符号在段中的偏移。</TD></TR>
" L* @3 B# e) o' O# W- }2 G<TR>" N8 Q, `/ ]- {
<TD vAlign=top width=55>STATIC</TD>
" N+ b1 Z& Q. j/ T5 @4 P<TD vAlign=top width=72>3</TD>
1 D, n' [. Z: `2 C<TD vAlign=top width=441>静态类型。ulValue为符号在段中的偏移。如果偏移为0,那么这个符号代表段名。</TD></TR>, R2 M0 i) x7 D7 W
<TR>
( \. i3 Y; u/ n( w2 j5 A/ N<TD vAlign=top width=55>REGISTER</TD>! G" [& R) V p' C0 p3 B. A
<TD vAlign=top width=72>4</TD>; x" B1 m/ h4 t3 Q) a+ j7 H( G
<TD vAlign=top width=441>寄存器变量。</TD></TR>
- L8 i# ]; F& g<TR>- Q$ {2 l. n2 ?) i
<TD vAlign=top width=55>MEMBER_OF_STRUCT</TD>8 {0 F& B9 o' `0 g& i$ s1 c
<TD vAlign=top width=72>8</TD>8 p K, `) e5 S# |0 ?2 E8 o" j" ]4 Y
<TD vAlign=top width=441>结构成员。ulValue值为该符号在结构中的顺序。</TD></TR>0 B6 I* S* \8 r5 n
<TR>8 a' {9 a, a- F3 |
<TD vAlign=top width=55>STRUCT_TAG</TD>
2 R6 V `" n" B! B<TD vAlign=top width=72>10</TD>: C8 v! ^; q' B4 U0 C
<TD vAlign=top width=441>结构标识符。</TD></TR>) s! n$ `- y; C( c/ D. ~
<TR>. r) O" L3 c+ [8 t% H: z" a5 p
<TD vAlign=top width=55>MEMBER_OF_UNION</TD>; P# ?# p9 l8 b. Q8 ~
<TD vAlign=top width=72>11</TD>: q5 G6 x& _: V9 J/ y
<TD vAlign=top width=441>联合成员。ulValue值为该符号在联合中的顺序。</TD></TR>: n4 k' y/ u) x4 K+ n7 u
<TR># G' J5 j" P* ^; H g) z1 t
<TD vAlign=top width=55>UNION_TAG</TD>3 a) C$ i2 O$ y* t! ?+ @
<TD vAlign=top width=72>12</TD>7 j6 B# O- c E6 t
<TD vAlign=top width=441>联合标识符。</TD></TR>! X7 K9 q& T% _/ O2 m8 X( a5 [
<TR>
: Y6 b/ B; X' _<TD vAlign=top width=55>TYPE_DEFINITION</TD>
. x' I, @* p( X<TD vAlign=top width=72>13</TD>
0 o8 U) r) ^/ Q, ~% z. C) C; }<TD vAlign=top width=441>类型定义。</TD></TR>
) A5 _7 C9 ~% ~, O<TR>
% B c' M* q/ N" _<TD vAlign=top width=55>FUNCTION</TD>& `2 }2 T" H5 |& \/ r2 f* s5 p
<TD vAlign=top width=72>101</TD>
* P- H: [9 Y6 m. m4 ^<TD vAlign=top width=441>函数名。</TD></TR>: W& |/ h# J5 L: f
<TR>; }" ^2 G1 E; _0 o( ^4 U7 H- ]
<TD vAlign=top width=55>FILE</TD>! l+ @7 j$ p' u5 I5 \
<TD vAlign=top width=72>102</TD>
, a& E8 d# |1 @0 K<TD vAlign=top width=441>文件名。</TD></TR></TABLE>, T' f' l! q* J- m4 V+ `
<DIV> 最后一个成员usNumAux是附加记录的数量。附加记录是用来描述符号的一些附加信息,为了便于保存,这些附加记录通常选择成为一条符号信息记录的整数倍(多数为1)。所以,如果这个成员的值为1,那么就表示在当前符号信息记录后附加了一条记录,用来保存附加信息。<br> 附加信息的结构是与符号的类型以及存储类型相关的。不同的类型的符号,其附加信息(如果有的话)的结构也不同。如果你不在意这些内容,也可以把它们乎略。<br> 当段的类型为FILE时,附加信息就是一个字符串,它是目标文件对应源文件的名称。其它类型在介绍PE时再进行详细讨论。</DIV>
g7 t6 a& {8 j! z! Q( Y[此贴子已经被作者于2005-9-17 14:53:44编辑过] |