PE文件(2)导出表

PE文件(2)导出表

导出表概念

导出表就是记载着动态链接库的一些导出信息。
通过导出表,DLL 文件可以向系统提供导出函数的名称、序号和入口地址等信息,Windows 加载器通过这些信息来完成动态连接的整个过程。

扩展名为.exe 的PE 文件中一般不存在导出表,而大部分的.dll 文件中都包含导出表。但这并不是绝对的。例如纯粹用作资源的.dll 文件就不需要导出函数啦,另外有些特殊功能的.exe 文件也会存在导出函数
当PE文件被加载为模块的时候,Windows加载器会将导入表中登记的DLL文件一并装入,再根据DLL文件中的函数导出信息对被执行文件的IAT表进行修正。
   Windows 在加载一个进程后就会在内存中为该进程开辟一个单独的4G虚拟地址空间(x86下)。有一些函数很多程序都会用到,为每一个程序所调用的相同的函数都占用一次内存空间显得很浪费,因此Windows提出了了动态链接库的概念,将一些常用的函数封装成动态链接库,等到需要的时候通过直接加载动态链接库,将需要的函数映射到自己的地址空间中,从而提高了内存的利用率。

定位导出表

导出表的定位总体上和导入表相似:

1.导出表的模块基地址的计算,与导入表相似,都是从PEB结构中得到的ImageBaseAddress成员。
与导入表不同的是,导入表是在exe模块中,而导出表是在dll模块中。

2.获得导出表的偏移是通过在NT头的扩展头的数据目录表的第IMAGE_DIRECTORY_ENTRY_EXPORT(1)项成员的VirtualAddress成员中。

只要将导入表中的IMAGE_NUMBEROF_ENTRY_IMPORT换为IMAGE_DIRECTORY_ENTRY_EXPORT就可以了。

    DWORD  v1 = 0;
    v1 = ImageNtHeaders.OptionalHeader.DataDirectory[IMAGE_DIRECTORY_ENTRY_EXPORT].VirtualAddress;

3.最后得到导出表的基地址:(ModuleBase + v1)

导出表结构

这里写图片描述

导出表的结构主要是两张表:函数名称地址表,入口地址表

typedef struct _IMAGE_EXPORT_DIRECTORY {
    DWORD   Characteristics;        //未使用,总是定义为0
    DWORD   TimeDateStamp;          //文件生成时间
    WORD    MajorVersion;
    WORD    MinorVersion;
    DWORD   Name;      //模块的真实名称RVA
    DWORD   Base;     //导出函数序号的起始值,将AddressOfFunctions 字段指向的入口地址表的索引号加上这个起始值就是对应函数的导出序号。假如Base 字段的值为x,那么入口地址表指定的第1个导出函数的序号就是x+0;第2个导出函数的序号就是x+1。总之,一个导出函数的导出序号等于Base 字段的值加上其在入口地址表中的位置索引值。
    DWORD   NumberOfFunctions;     //导出函数的个数
    DWORD   NumberOfNames;    //名称方式导出的函数的总数。有的导出函数是没有名字的,只有序号
    DWORD   AddressOfFunctions;     //一个RVA 值,指向包含全部导出函数入口地址的双字数组(EAT)。数组中的每一项是一个RVA 值(函数入口地址),数组的项数等于NumberOfFunctions 字段的值。
    DWORD   AddressOfNames;         //一个RVA 值,指向函数名的字符串地址表。这个地址表是一个双字数组,数组中的每一项指向一个函数名称字符串的RVA。数组的项数等NumberOfNames 字段的值,所有有名称的导出函数的名称字符串都定义在这个表中。
    DWORD   AddressOfNameOrdinals;  //一个RVA 值,指向另一个word 类型的数组(注意不是双字数组)。数组项目与文件名地址表中的项目一一对应,项目值代表函数入口地址表的索引,这样函数名称与函数入口地址关联起来,起到一个桥梁的作用。
} IMAGE_EXPORT_DIRECTORY, *PIMAGE_EXPORT_DIRECTORY

导出表遍历

导出表遍历的两种方法:
(1)通过函数序号得到函数地址:   
    1>由DOS头定位到PE头。
    2>由PE头定位到PE头中的中的 可选头IMAGE_OPTIONAL_HEADER32 结构中取出数据目录表,并从第一个数据目录中得到导出表的RVA。
    3>从导出表的 Base 字段得到起始序号。
    4>将需要查找的导出序号减去起始序号,得到函数在入口地址表中的索引。
    5>检测索引值是否大于导出表的 NumberOfFunctions 字段的值,如果大于后者的话,说明输入的序号是无效的。
    6>用这个索引值在 AddressOfFunctions 字段指向的导出函数入口地址表中取出相应的项目,这就是函数入口地址的RVA 值,当函数被装入内存的时候,这个RVA 值加上模块实际装入的基地址,就得到函数真正的入口地址。
 
(2)通过函数名称得到函数地址:
    1>由DOS头定位到PE头。(同上)
    2>由PE头定位到PE头中的中的 可选头IMAGE_OPTIONAL_HEADER32 结构中取出数据目录表,并从第一个数据目录中得到导出表的RVA。(同上)   
    3>从导出表的NumberOfNames 字段得到已命名函数的总数,并以这个数字作为循环的次数来构造一个循环。
    4>从AddressOfNames 字段指向得到的函数名称地址表的第一项开始,在循环中将每一项定义的函数名与要查找的函数名相比较,如果没有任何一个函数名是符合的,表示文件中没有指定名称的函数。
    5>如果某一项定义的函数名与要查找的函数名符合,那么记下这个函数名在字符串地址表中的索引值,然后在AddressOfNamesOrdinals 指向的数组中以同样的索引值取出数组项的值,我们这里假设这个值是x。
    6>以 x 值作为索引值,在 AddressOfFunctions 字段指向的函数入口地址表中获取的 RVA 加上模块基地址就是函数的入口地址。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值