导出表 IMAGE_EXPORT_DIRECTORY

typedef struct _IMAGE_EXPORT_DIRECTORY {
    DWORD   Characteristics;    // 未使用,总为0     DWORD   TimeDateStamp;      // 文件创建时间戳
    WORD    MajorVersion;       // 未使用,总为0     WORD    MinorVersion;       // 未使用,总为0
    DWORD   Name;               // 指向一个代表此 DLL名字的 ASCII字符串的 RVA
    DWORD   Base;               // 函数的起始序号
    DWORD   NumberOfFunctions;  // 导出函数的总数    DWORD   NumberOfNames;      // 以名称方式导出的函数的总数    DWORD   AddressOfFunctions;     // 指向输出函数地址的RVA
    DWORD   AddressOfNames;         // 指向输出函数名字的RVA
    DWORD   AddressOfNameOrdinals;  // 指向输出函数序号的RVA} IMAGE_EXPORT_DIRECTORY, *PIMAGE_EXPORT_DIRECTORY;

AddressOfFunctions 所指向内容是以 4 字节为一个单位的数组元素,每个元素代表函数入口

AddressOfNames 所指向内容是以 4 字节为一个单位的数组元素,每个元素代表一个指向字符串的 RVA

AddressOfNamesOrdinals 所指向内容是以 2 字节为一个单位的数组元素,每个元素代表对应名字在 AddressOfFunctions 中的序号数。

AddressOfNamesAddressOfNamesOrdinals 的数目肯定是一样的,不是一样那么就出错了。

主要要掌握两种寻找函数入口地址的方法:
A. 从序号查找函数入口地址

  1. 定位到PE 文件头

  2. 从PE 文件头中的 IMAGE_OPTIONAL_HEADER32 结构中取出数据目录表,并从第一个数据目录中得到导出表的RVA

  3. 从导出表的 Base 字段得到起始序号

  4. 将需要查找的导出序号减去起始序号Base,得到函数在入口地址表中的索引,检测索引值是否大于导出表的 NumberOfFunctions 字段的值,如果大于后者的话,说明输入的序号是无效的

  5. 用这个索引值在 AddressOfFunctions 字段指向的导出函数入口地址表中取出相应的项目,这就是函数入口地址的RVA 值,当函数被装入内存的时候,这个RVA 值加上模块实际装入的基地址,就得到了函数真正的入口地址
    B. 从函数名称查找入口地址
    我想通的地方,记录下来:用函数名来查找的话,Base 的值现在没有任何意义

  6. 首先得到导出表的地址

  7. 从导出表的 NumberOfNames 字段得到已命名函数的总数,并以这个数字作为循环的次数来构造一个循环,从 AddressOfNames 字段指向得到的函数名称地址表的第一项开始,在循环中将每一项定义的函数名与要查找的函数名相比较,如果没有任何一个函数名是符合的,表示文件中没有指定名称的函数。

  8. 如果某一项定义的函数名与要查找的函数名符合,那么记下这个函数名在字符串地址表中的索引值,然后在AddressOfNamesOrdinals 指向的数组中以同样的索引值取出数组项的值,我们这里假设这个值是 x

  9. 最后,以 x 的值作为索引值在 AddressOfFunctions 字段指向的函数入口地址表中获取 RVA 。此 RVA 就是函数的入口地址。

附上图片:
在这里插入图片描述

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
导出是一个二进制文件,内部存储了函数名和函数地址的对应关系。在 Windows 系统中,导出通常存储在 DLL 动态链接库中,供程序在运行时调用。 如果想查看 MemoryProfilerRuntimeWin.dll 的导出,可以使用 Visual Studio 自带的工具 dumpbin.exe。具体步骤如下: 1. 打开 Visual Studio 开发人员命令提示符,输入以下命令: ``` dumpbin /exports MemoryProfilerRuntimeWin.dll ``` 2. 运行命令后,会显示 MemoryProfilerRuntimeWin.dll 的导出信息,包括函数名和函数地址。 如果需要在代码中获取导出信息,可以使用 Win32 API 函数 EnumExports。具体使用方法可以参考以下代码: ``` HMODULE hModule = LoadLibrary(TEXT("MemoryProfilerRuntimeWin.dll")); if (hModule != NULL) { // 获取导出信息 DWORD dwSize = 0; PIMAGE_EXPORT_DIRECTORY pExportDir = (PIMAGE_EXPORT_DIRECTORY)ImageDirectoryEntryToData(hModule, TRUE, IMAGE_DIRECTORY_ENTRY_EXPORT, &dwSize); if (pExportDir != NULL) { PDWORD pdwFuncName = (PDWORD)((DWORD)pExportDir + pExportDir->AddressOfNames); PDWORD pdwFuncAddr = (PDWORD)((DWORD)pExportDir + pExportDir->AddressOfFunctions); PWORD pwFuncOrd = (PWORD)((DWORD)pExportDir + pExportDir->AddressOfNameOrdinals); for (DWORD i = 0; i < pExportDir->NumberOfNames; i++) { // 获取函数名和函数地址 char* pszFuncName = (char*)((DWORD)hModule + pdwFuncName[i]); DWORD dwFuncAddr = (DWORD)hModule + pdwFuncAddr[pwFuncOrd[i]]; printf("%s: %x\n", pszFuncName, dwFuncAddr); } } FreeLibrary(hModule); } ``` 以上代码会依次输出 MemoryProfilerRuntimeWin.dll 中的所有导出函数名和函数地址。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值