PE文件结构详解之头信息解析

2024-06-03 23:12

本文主要是介绍PE文件结构详解之头信息解析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

PE文件结构详解

  • 一、前言
    • 1.概述
    • 2.PE文件结构
    • 3.所用工具
  • 二、DOS头(DOS Header)解析
    • 1.作用
    • 2.图例
    • 3.参数详解
    • 4.总结
  • 三、DOS Stub
    • 1.作用
    • 2.图例
  • 四、NT头(NT Header)解析
    • 1.作用
    • 2.PE标识图例
    • 3.文件头(COFF头)图例
    • 4.可选头(Optional Header)图例
  • 五、区段头(Section Header)
    • 1.作用
    • 2.图例
  • 六、附C++解析源码

一、前言

1.概述

PE文件(Portable Executable File)是Windows上最常见的可执行文件,按文件后缀来说就是.exe.dll文件,还有一些其他的文件,例如.sys系统文件,不过最常见以及常用的就是.exe.dll,在初学阶段狭义上也可以就把PE文件就理解成.exe和.dll文件。

2.PE文件结构

  1. DOS头(DOS Header)
  2. DOS Stub
  3. NT头(NT Header)
    • PE标识(Signature)
    • 文件头(File Header)
    • 可选头(OptionHeader)
  4. 区段头(Section Header)

3.所用工具

WinHex-20.7-x86-x64.exe
PETool v1.0.0.5.exe
010EditorProtable


二、DOS头(DOS Header)解析

1.作用

  1. 兼容性: 让老的DOS系统识别这是一个可执行文件,即使它不能运行。
  2. 定位PE头: e_lfanew字段指向PE头的开始位置,操作系统通过这个字段找到PE文件的真正头部,从而加载和执行文件。

2.图例

范围:起始地址开始,长度64字节
我们可以通过使用WinHex软件来打开一个PE文件,其中如下图红框包裹的部分就是DOS头的内容,长度固定为64个字节。
在这里插入图片描述

3.参数详解

其中注释的内容不重要,可以忽略。

typedef struct _IMAGE_DOS_HEADER {      // DOS .EXE头WORD   e_magic;                     // 魔数(Magic number),固定MZ//WORD   e_cblp;                      // 文件最后一页的字节数//WORD   e_cp;                        // 文件中的页数//WORD   e_crlc;                      // 重定位项数目//WORD   e_cparhdr;                   // 头部大小,以段落(16字节)为单位//WORD   e_minalloc;                  // 程序所需的最小额外段数//WORD   e_maxalloc;                  // 程序所需的最大额外段数//WORD   e_ss;                        // 初始(相对)SS值//WORD   e_sp;                        // 初始SP值//WORD   e_csum;                      // 校验和//WORD   e_ip;                        // 初始IP值//WORD   e_cs;                        // 初始(相对)CS值//WORD   e_lfarlc;                    // 重定位表的文件地址//WORD   e_ovno;                      // 覆盖编号//WORD   e_res[4];                    // 保留字//WORD   e_oemid;                     // OEM标识符(用于e_oeminfo)//WORD   e_oeminfo;                   // OEM信息(由e_oemid指定)//WORD   e_res2[10];                  // 保留字LONG   e_lfanew;                    // 新EXE头的文件地址(PE头的偏移量)
} IMAGE_DOS_HEADER, *PIMAGE_DOS_HEADER;

4.总结

其中的e_magic参数可以用来判断这个文件是不是PE文件。
e_lfanew表示的是新的PE头的偏移位置,例如程序的起始地址是0x01,e_lfanew的值是0xF0,那么新的PE头的位置就是0x01+0xF0。


三、DOS Stub

1.作用

一个典型的DOS Stub可能会包含一个简短的DOS程序,这个程序通常会执行以下操作:

  1. 显示一条消息,说明该程序需要在Windows环境下运行。
  2. 终止程序的执行。

约等于没有用,因为DOS Stub在windwos系统上是不会执行的。

2.图例

范围:DOS头后开始,至e_lfanew偏移量结束。
我们可以看到红框的右边解析的Ascii码,里面有这么一段文字。
This program cannot be run in DOS mode.
可得出结论这玩意在windows上没什么用,就是在DOS系统上显示消息用的。
范围为之后 至 e_lfanew偏移量之前。
在这里插入图片描述


四、NT头(NT Header)解析

1.作用

NT头,其中包含三个部分

  1. PE标识(PE签名,Signature)
  2. 标准文件头(COFF头,Common Object File Format Header)
  3. 可选头(Optional Header)

PE头就是我们PE文件的最重要的部分之一了,其中包含了很多重要的信息。

2.PE标识图例

范围:e_lfanew偏移量开始,长度4字节。
其中数据0x00004550(从小到大读)就是:
50=P
45=E
00=\0
00=\0
在这里插入图片描述

3.文件头(COFF头)图例

范围:PE标识开始,长度20字节
在这里插入图片描述
注释的内容不重要,可以忽略。

typedef struct _IMAGE_FILE_HEADER {WORD  Machine;              // 指定目标机器类型WORD  NumberOfSections;     // 文件中的节数//DWORD TimeDateStamp;        // 文件创建的时间戳//DWORD PointerToSymbolTable; // 指向符号表的指针(通常为0)//DWORD NumberOfSymbols;      // 符号表中的符号数(通常为0)WORD  SizeOfOptionalHeader; // 可选头的大小WORD  Characteristics;      // 文件的属性标志
} IMAGE_FILE_HEADER, *PIMAGE_FILE_HEADER;

4.可选头(Optional Header)图例

文件头开始,长度为文件头的SizeOfOptionalHeader属性
在这里插入图片描述
注释的内容不重要,可以忽略。
其中的DllCharacteristics中有一条DYNAMIC_BASE表示是否动态基址,可以用010 editor来查看这个值,如下图。
在这里插入图片描述

typedef struct _IMAGE_OPTIONAL_HEADER {WORD    Magic;                       // 标识文件类型,0x10B表示PE32,0x20B标识PE64
//    BYTE    MajorLinkerVersion;          // 链接器的主版本号
//    BYTE    MinorLinkerVersion;          // 链接器的次版本号
//    DWORD   SizeOfCode;                  // 所有代码节的总大小
//    DWORD   SizeOfInitializedData;       // 所有已初始化数据节的总大小
//    DWORD   SizeOfUninitializedData;     // 所有未初始化数据节的总大小DWORD   AddressOfEntryPoint;         // 程序入口点的地址(RVA)OEP
//    DWORD   BaseOfCode;                  // 代码节的起始地址(RVA)
//    DWORD   BaseOfData;                  // 数据节的起始地址(RVA)DWORD   ImageBase;                   // 首选的加载地址DWORD   SectionAlignment;            // 内存对齐大小DWORD   FileAlignment;               // 文件对齐大小
//    WORD    MajorOperatingSystemVersion; // 操作系统的主版本号
//    WORD    MinorOperatingSystemVersion; // 操作系统的次版本号
//    WORD    MajorImageVersion;           // 映像文件的主版本号
//    WORD    MinorImageVersion;           // 映像文件的次版本号
//    WORD    MajorSubsystemVersion;       // 子系统的主版本号
//    WORD    MinorSubsystemVersion;       // 子系统的次版本号
//    DWORD   Win32VersionValue;           // 保留字段,应为0DWORD   SizeOfImage;                 // 文件在内存中的大小,按照SectionAlignment对齐后DWORD   SizeOfHeaders;               // 所有头和节表(区段头)的总大小,按照FileAlignment对齐后
//    DWORD   CheckSum;                    // 校验和
//    WORD    Subsystem;                   // 子系统类型
//    WORD    DllCharacteristics;          // DLL的特性
//    DWORD   SizeOfStackReserve;          // 保留的栈大小
//    DWORD   SizeOfStackCommit;           // 初始提交的栈大小
//    DWORD   SizeOfHeapReserve;           // 保留的堆大小
//    DWORD   SizeOfHeapCommit;            // 初始提交的堆大小
//    DWORD   LoaderFlags;                 // 加载器标志,应为0DWORD   NumberOfRvaAndSizes;         // 数据目录的数量IMAGE_DATA_DIRECTORY DataDirectory[IMAGE_NUMBEROF_DIRECTORY_ENTRIES]; // 数据目录数组
} IMAGE_OPTIONAL_HEADER32, *PIMAGE_OPTIONAL_HEADER32;

五、区段头(Section Header)

1.作用

区段也成为“节”,区段头也叫节表
注释的不重要,可以忽略

2.图例

位置:可选头开始,区段头多个的,每个的固定大小为40个字节,区段头的数量存放在标准头的NumerOfSections属性

typedef struct _IMAGE_SECTION_HEADER {BYTE  Name[IMAGE_SIZEOF_SHORT_NAME];    // 节的名称,通常是一个8字节长的字符串,如“.text”、“.data”等
//    union {
//        DWORD PhysicalAddress;              // 物理地址,不常用
//        DWORD VirtualSize;                  // 节在内存中的实际大小
//    } Misc;DWORD VirtualAddress;                   // 区段在内存中的偏移位值
//    DWORD SizeOfRawData;                    // 区段在文件中对齐后的大小,文件对齐(File Alignment)后的大小DWORD PointerToRawData;                 // 区段在文件中的偏移值
//    DWORD PointerToRelocations;             // 重定位信息表在文件中的位置偏移,通常为0
//    DWORD PointerToLinenumbers;             // 行号信息在文件中的位置偏移,调试信息相关,通常为0
//    WORD  NumberOfRelocations;              // 重定位项的数量
//    WORD  NumberOfLinenumbers;              // 行号信息的数量DWORD Characteristics;                  // 节的属性标志,描述节的特性(可执行、可读、可写等)
} IMAGE_SECTION_HEADER, *PIMAGE_SECTION_HEADER; 

六、附C++解析源码

C++解析PE文件源码github地址

这篇关于PE文件结构详解之头信息解析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

网页解析 lxml 库--实战

lxml库使用流程 lxml 是 Python 的第三方解析库,完全使用 Python 语言编写,它对 XPath表达式提供了良好的支 持,因此能够了高效地解析 HTML/XML 文档。本节讲解如何通过 lxml 库解析 HTML 文档。 pip install lxml lxm| 库提供了一个 etree 模块,该模块专门用来解析 HTML/XML 文档,下面来介绍一下 lxml 库

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

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

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

usaco 1.3 Mixing Milk (结构体排序 qsort) and hdu 2020(sort)

到了这题学会了结构体排序 于是回去修改了 1.2 milking cows 的算法~ 结构体排序核心: 1.结构体定义 struct Milk{int price;int milks;}milk[5000]; 2.自定义的比较函数,若返回值为正,qsort 函数判定a>b ;为负,a<b;为0,a==b; int milkcmp(const void *va,c

业务中14个需要进行A/B测试的时刻[信息图]

在本指南中,我们将全面了解有关 A/B测试 的所有内容。 我们将介绍不同类型的A/B测试,如何有效地规划和启动测试,如何评估测试是否成功,您应该关注哪些指标,多年来我们发现的常见错误等等。 什么是A/B测试? A/B测试(有时称为“分割测试”)是一种实验类型,其中您创建两种或多种内容变体——如登录页面、电子邮件或广告——并将它们显示给不同的受众群体,以查看哪一种效果最好。 本质上,A/B测

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

【北交大信息所AI-Max2】使用方法

BJTU信息所集群AI_MAX2使用方法 使用的前提是预约到相应的算力卡,拥有登录权限的账号密码,一般为导师组共用一个。 有浏览器、ssh工具就可以。 1.新建集群Terminal 浏览器登陆10.126.62.75 (如果是1集群把75改成66) 交互式开发 执行器选Terminal 密码随便设一个(需记住) 工作空间:私有数据、全部文件 加速器选GeForce_RTX_2080_Ti

K8S(Kubernetes)开源的容器编排平台安装步骤详解

K8S(Kubernetes)是一个开源的容器编排平台,用于自动化部署、扩展和管理容器化应用程序。以下是K8S容器编排平台的安装步骤、使用方式及特点的概述: 安装步骤: 安装Docker:K8S需要基于Docker来运行容器化应用程序。首先要在所有节点上安装Docker引擎。 安装Kubernetes Master:在集群中选择一台主机作为Master节点,安装K8S的控制平面组件,如AP

自定义类型:结构体(续)

目录 一. 结构体的内存对齐 1.1 为什么存在内存对齐? 1.2 修改默认对齐数 二. 结构体传参 三. 结构体实现位段 一. 结构体的内存对齐 在前面的文章里我们已经讲过一部分的内存对齐的知识,并举出了两个例子,我们再举出两个例子继续说明: struct S3{double a;int b;char c;};int mian(){printf("%zd\n",s