PE之FOA与RVA互相转换过程与C语言实现

2023-10-12 02:40

本文主要是介绍PE之FOA与RVA互相转换过程与C语言实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 说明
  • 一、FOA和RVA
  • 二、RVA转为FOA
    • 1.大致步骤
    • 2.特殊情况
    • 3.C语言实现函数功能
  • 三、FOA转为RVA
    • 1.大致步骤
    • 2.特殊情况
    • 3.C语言实现函数功能

说明

看滴水的视频写学习笔记总结

语言:c/c++

编译环境:vc++6.0

C语言函数中定义的结构类型来自于头文件windows.h

准确的说,定义的PE的结构体类型的所有数据都来自与头文件winnt.h,只不过windows.h内部声明了winnt.h

一、FOA和RVA

缩写英文全称含义
FOAFile Offset Address文件偏移地址
RVARelative Virtual Addresses相对虚拟地址
VAVirtual Address虚拟地址

RVA

VA = RVA + ImageBase

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-d6wchHYO-1590056089133)(E:\Typora\image\image-20200521172839449.png)]

二、RVA转为FOA

1.大致步骤

第一步:判断RVA是否在PE头区。在PE头区RVA与FOA相等,直接返回RVA。

对比第一个节表头VirtualAddress.
若RVA < VirtualAddress则说明RVA在PE头区

第二步:若RVA没有在头区,就遍历节表头,寻找这个RVA位于哪个节表区。

循环遍历节表头,在该 节表头内 满足下列条件
VirtualAddress <= RVA <= VirtualAddress + SizeOfRawData
即可确定RVA位于哪个节表头

第三步:找到RVA对应的节表区头后,计算并返回FOA的值

在对应的节表头区内,
FOA = (RVA - VirtualAddress) + PointerToRawData;

2.特殊情况

两种情况的RVA没有对应的FOA

  • RVA为内存拉伸后系统填充的地址

  • RVA超出内存范围

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-skJxECmP-1590056089136)(E:\Typora\image\image-20200521174958298.png)]

3.C语言实现函数功能

说明意义
函数名称size_t ConvertRvaToFoa( size_t RVA , LPVOID pFileBuffer )
功能将RVA转化为FOA
参数参数1:要转换的RVA的值(size_t);参数2:指向文件缓冲区的指针(LPVOID)
返回值成功则返回对应的FOA,失败则返回0(这个RVA是没有对应的FOA,为内存拉伸后系统填充的地址,或者RVA超出内存范围)
调用头文件stdio.h window.h
//功能:将RVA转化为FOA
//参数:参数1:要转换的RVA的值(size_t);参数2:指向文件缓冲区的指针(LPVOID)
//返回值:成功则返回对应的FOA,失败则返回0(这个RVA是没有对应的FOA,为内存拉伸后系统填充的地址,或者RVA超出内存范围)。
size_t ConvertRvaToFoa( size_t RVA , LPVOID pFileBuffer )
{int i ;//用于遍历节表
//	size_t FOA = 0;//定义表头指针PIMAGE_DOS_HEADER pDosHeader = NULL;PIMAGE_NT_HEADERS32 pNtHeader = NULL;PIMAGE_FILE_HEADER pFileHeader = NULL;PIMAGE_OPTIONAL_HEADER pOptionHeader = NULL;PIMAGE_SECTION_HEADER pSectionHeader = NULL;	//给表头赋初值pDosHeader = (PIMAGE_DOS_HEADER)pFileBuffer;pNtHeader = (PIMAGE_NT_HEADERS32)((DWORD)pDosHeader + pDosHeader->e_lfanew);pFileHeader = (PIMAGE_FILE_HEADER)( (DWORD)pNtHeader+4 );pOptionHeader = (PIMAGE_OPTIONAL_HEADER)( (DWORD)pFileHeader + IMAGE_SIZEOF_FILE_HEADER);	//第一个节表头pSectionHeader = (PIMAGE_SECTION_HEADER)( (DWORD)pOptionHeader + pFileHeader->SizeOfOptionalHeader);if(RVA < pSectionHeader->VirtualAddress)//判断RVA是否在PE头区{if(RVA < pSectionHeader->PointerToRawData)return RVA;//此时FOA == RVAelsereturn 0;}for(i=0 ; i<pFileHeader->NumberOfSections ; i++)//循环遍历节表头{if( i )//遍历节表头,第一次不遍历,pSectionHeader = (PIMAGE_SECTION_HEADER)( (DWORD)pSectionHeader + IMAGE_SIZEOF_SECTION_HEADER );if(RVA >= pSectionHeader->VirtualAddress )//是否大于这个节表的RVA{if( RVA <= pSectionHeader->VirtualAddress + pSectionHeader->SizeOfRawData )//判断是否在这个节区return ( RVA - pSectionHeader->VirtualAddress ) + pSectionHeader->PointerToRawData;//确定节区后,计算FOA}else//RVA不可能此时的pSectionHeader->VirtuallAddress小,除非是返回值为0的情况。return 0;		}return 0;
}

三、FOA转为RVA

1.大致步骤

VA转为FOA的步骤框架差不多,但细节不一样

第一步:判断FOA是否在PE头区。在PE头区FOA与RVA相等,直接返回这个FOA。

对比第一个节表头的PointerToRawData
若FOA < PointerToRawData则证明FOA在PE头区

第二步:若FOA没有在头区,就遍历节表头,寻找这个FOA位于哪个节表区。

循环遍历节表头,在该 节表头内 满足下列条件
PointerToRawData <= FOA < PointerToRawData + SizeOfRawData
即可确定FOA位于哪个节表头

第三步:找到FOA对应的节表区头后,计算并返回RVA的值

在对应的节表头区内,
RVA = (FOA - PointerToRawData) + VirtualAddress;

2.特殊情况

FOA超出文件内存的范围

3.C语言实现函数功能

说明意义
函数名称size_t ConvertFoaToRva( size_t FOA , LPVOID pFileBuffer )
功能将FOA转换为RVA
参数参数1:要转换的FOA值(size_t);参数2:指向文件内存的指针(LPVOID)
返回值成功则返回对应的RVA,失败返回0(此时为FOA越界)
调用头文件stdio.h windows.h
//功能:将FOA转换为RVA
//参数:参数1:要转换的FOA值(size_t);参数2:指向文件内存的指针(LPVOID);
//返回值:成功则返回对应的RVA,失败返回0(此时为FOA越界)。
size_t ConvertFoaToRva( size_t FOA , LPVOID pFileBuffer )
{int i = 0;//用于遍历节表。
//	size_t RVA = 0;//定义表头指针PIMAGE_DOS_HEADER pDosHeader = NULL;PIMAGE_NT_HEADERS32 pNtHeader = NULL;PIMAGE_FILE_HEADER pFileHeader = NULL;PIMAGE_OPTIONAL_HEADER pOptionHeader = NULL;PIMAGE_SECTION_HEADER pSectionHeader = NULL;//给表头赋初值pDosHeader = (PIMAGE_DOS_HEADER)pFileBuffer;pNtHeader = (PIMAGE_NT_HEADERS32)( (DWORD)pDosHeader + pDosHeader->e_lfanew  );pFileHeader = (PIMAGE_FILE_HEADER)( (DWORD)pNtHeader + 4 );pOptionHeader = (PIMAGE_OPTIONAL_HEADER)( (DWORD)pFileHeader + IMAGE_SIZEOF_FILE_HEADER);//第一个节表头pSectionHeader = (PIMAGE_SECTION_HEADER)( (DWORD)pOptionHeader + pFileHeader->SizeOfOptionalHeader );if( FOA < pSectionHeader->PointerToRawData )//判断是否位于 头区return FOA ; //这是RVA == FOA ;for(i=0 ; i<pFileHeader->NumberOfSections ; i++)//循环遍历节表头{if( i )//遍历节表头,第一次不遍历,pSectionHeader = (PIMAGE_SECTION_HEADER)( (DWORD)pSectionHeader + IMAGE_SIZEOF_SECTION_HEADER);if( FOA >= pSectionHeader->PointerToRawData )//是否大于这个节表的FOA{if( FOA < pSectionHeader->PointerToRawData + pSectionHeader->SizeOfRawData )//判断是否在这个节表区域return (FOA - pSectionHeader->PointerToRawData ) + pSectionHeader->VirtualAddress ;//计算并返回RVA}			}return 0;
}

欢迎大家留言交流 ^ _ ^

这篇关于PE之FOA与RVA互相转换过程与C语言实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/192663

相关文章

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

作业提交过程之HDFSMapReduce

作业提交全过程详解 (1)作业提交 第1步:Client调用job.waitForCompletion方法,向整个集群提交MapReduce作业。 第2步:Client向RM申请一个作业id。 第3步:RM给Client返回该job资源的提交路径和作业id。 第4步:Client提交jar包、切片信息和配置文件到指定的资源提交路径。 第5步:Client提交完资源后,向RM申请运行MrAp

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss