二进制分析 ELF格式

简介: ELF 用于可执行文件,对象文件,共享库及核心转储,是UN IX系统实验室作为二进制接口而开发和发布

组成部分
ELF 头部

该头部是一系列结构化的字节,描述是什么样的二进制文件及在文件的什么地方找到其他内容

可以在 /usr/include/elf.h 或规范中找到其它内型的定义

程序头描述的是一个段在文件中的位置,大小以及它被放进内存后所在的位置和大小
二进制文件中的代码和数据在逻辑上被分为连续的非重叠块
节头指定节的属性,并允许找到节中字节的位置

ELF 头部结构
unsigned char e_ident[16]

/* Magic number and other info */

[0~3]: 幻数,0x7f 说明在处理一个ELF 文件

[4]: EI_CLASS,系统类型,表示用32位还是64位

[5]: EI_DATA,字节顺序(大端还是小段)

[6]: EI_VERSION,使用ELF 规范版本 

[7]: EI_OSABI ,二进制接口,非零表示文件中会使用一些ABI 或OS 的具体扩展名

[8]: EI_ABIVERSION,表示EI_OSABI 字节指定的ABI版本 

[9~15]: EI_PAD,保留字节,供将来使用

uint16_t e_type

 /* Object file type */

指定文件类型,

        ET_REL: 可重定位对象文件

        ET_EXEC: 可执行的二进制文件

        ET_DYN: 动态库

uint16_t e_machine

/* Architecture */

运行架构体系,

        EM_X86_64: 64位x86

        EM_386: 32位x86

        EM_ARM: ARM二进制文件

uint32_t e_versioin

/* Object file version */

使用的ELF 规范版本

uint64_t e_entry

 /* Entry point virtual address */

二进制文件的入口点,应该执行的虚拟地址 

uint64_t phoff

 /* Program header table file offset */

程序表距离开始的偏移量

uint64_t e_shoff

  /* Section header table file offset */

节头表距离开始的偏移量

uint32_t e_flags

/* Processor-specific flags */

处理器标志,ARM,X86,...

uint16_t e_ehsize

/* ELF header size in bytes */

ELF 头部大小

uint16_t e_phentsize

/* Program header table entry size */

程序头表大小

uint16_t e_phum

 /* Program header table entry count */

程序头表程序头数量

uint16_t e_shentsize

/* Section header table entry size */

节头表大小

uint16_t e_shum

 /* Section header table entry count */

节头表节的数量

uint16_t e_eshstrndx

/* Section header string table index */

表示字符串表在节头表中的索引

程序头结构 Elf64_Phdr
uint32_Word  p_type

 /* Segment type */

标识段类型:

        PT_LOAD: 在创建进程时加载到内存中

        PT_DYNAMIC:包含.dynamic 节,告诉解析器如果解析二进制文件用于执行

        PT_INTERT:包含了.interp节该节提供了加载二进制文件的解析器的名称

        ...

uint32_Word  p_flags;  

/* Segment flags */

指定段子啊运行时的访问权限

        PF_X:该段可执行

        PF_W: 该段可写

        PF_R:可读

uint64_Off   p_offset;

 /* Segment file offset */

段起始文件便宜量

uint64_Addr  p_vaddr;

 /* Segment virtual address */

加载的虚拟地址

 uint64_Addr  p_paddr;    

/* Segment physical address */

加载的物理地址,在某些系统上,可以指定段在物理内存的那个地址上执行

uint64_Xword p_filesz; 

 /* Segment size in file */

段大小

uint64_Xword p_memsz;    

 /* Segment size in memory */

内存大小

 uint64_Xword p_align;

 /* Segment alignment */

指定段所需要的内存对齐方式

节头Elf64_Shdr
uint32_Word   sh_name;

/* Section name (string tbl index) */

名字索引,如果为0,表示该节没有索引

init,fini,text,.....

uint32_Word   sh_type; 

  /* Section type */

类型:

        SHT_PROGBITS: 包含了程序数据

        SHT_SYSMATA:  静态符号表

        SHT_DYNSYM: 动态链接器符合表

        SHT_STRTAB: 字符串标

        SHT_REL: 

        SHT_REAL

        ....

uint64_Xword  sh_flags;

/* Section flags */

节标志:

        SHF_WRITE: 指示该节在运行时可写

        SHF_ALLOC: 指示执行二进制文件时讲节的内容加载到虚拟内存中
        SHF_EXECINSTR: 指示该节包含可执行指令

uint64_Addr   sh_addr;  

 /* Section virtual addr at execution */

该节的虚拟地址

uint64_Off    sh_offset;

 /* Section file offset */

该节的文件便宜量

 uint64_Xword  sh_size;

 /* Section size in bytes */

节的大小

uint64_Word   sh_link;

/* Link to another section */

链接到下一个节

 uint64_Word   sh_info; 

/* Additional section information */

存放关于节的额外信息,依赖于节的类型

uint64_Xword  sh_addralign;

/* Section alignment */

在内存中的对齐方式

 uint64_Xword  sh_entsize;

/* Entry size if section holds table */

某些节包含固定大小的条目,该字段指定每个条目的长度字节数,如果为0则并不包含固定长度条目的表格

名称类型说明
.interpPROGBITS

interp字段的值是一个字符串指针,指向文件头中的interp_name字段,用于指定解释器或加载器的名称

例如,如果ELF文件是为C++程序编译的,那么interp_name字段可能包含“ld-linux-x86-64.so.2”的字符串,表示GNU linker(ld)程序。

.note.ABI-tagNOTE

用于存储文件的应用二进制接口(ABI)信息。ABI是一个程序的接口定义,用于描述程序的可执行代码和库之间的交互方式。

包含了文件的ABI版本号、ABI类型(例如,32位或64位)以及其他相关信息

.note.gnu.build-idNOTE

用于存储文件的唯一标识符(build ID)。build ID是一个字符串,用于标识文件的构建过程和源代码。

用于标识文件的构建过程和源代码。它用于帮助诊断和调试问题,因为它可以帮助跟踪文件的来源和构建过程。

.gnu.hashGNU_HASH

用于存储文件的哈希值。哈希值是一个字符串,用于标识文件的内容和结构。

通常由文件的构建过程生成。这个字符串通常包含文件的哈希值,用于帮助诊断和调试问题,因为它可以帮助跟踪文件的内容和结构。

.dynsymDYNSYM

用于存储动态符号表(dynsym)信息。动态符号表是一个数据结构,用于描述程序中的动态符号(例如,库和其他程序中的函数和变量)。

用于帮助程序在运行时查找和链接动态库中的符号。

.dynstrSTRTAB用于存储动态字符串表(dynstr)信息。动态字符串表是一个数据结构,用于存储程序中的动态字符串(例如,库和其他程序中的函数和变量名称)。
.gun.versionVERSYM用于存储文件的GNU版本信息。GNU版本信息是一个字符串,用于标识文件的GNU工具链版本。
.gun.version_rVERNEED用于存储文件的GNU运行时版本信息。GNU运行时版本信息是一个字符串,用于标识文件的GNU运行时库版本。
.rela.dynRELA记录所有变量的动态链接器重定位信息与rela.plt 一起使用
.rela.pltRELA过程的动态链接重定位表
.initPROGBITS在执行其他代码之前的初始化(类似构造函数)
.pltPROGBITS过程链接表(用于延迟绑定:在运行中需要的时候执行)
.plt.gotPROGBITS全局偏移量表(用于延迟绑定)
.textPROGBITS主要代码,包含用户代码,注意节的标志,一般来说可执行就不可写,可写就不可执行,因为可执行又可写就会被攻击者利用,直接覆盖代码来修改程序
.finiPROGBITS主程序运行完成后执行的代码(类似析造函数)
.rodataPROGBITS只读数据,用于存储常量
.en_frame_hdrPROGBITS用于存储程序的执行框架头部信息。执行框架头部是一个数据结构,用于描述程序的执行框架和运行时状态。
.en_framePROGBITS用于存储程序的执行框架信息。执行框架是一个数据结构,用于描述程序的执行框架和运行时状态。
.init_arrayINIT_ARRAY包含一个指向构造函数的指针数组,在main 之前会被依次调用,与init 不同的时它是数据节(指针可被修改,使其方便插入钩子的位置)
.fini_arrayFINT_ARRAY包含指向析构函数数组指针(指针可被修改,使其方便插入钩子的位置)
.jcrPROGBITS
.dynamicDYNAMIC用于存储文件的动态链接信息。动态链接信息是一个数据结构,用于描述程序中的动态链接(例如,库和其他程序中的函数和变量)。
.gotPROGBITS用于存储文件的全局偏移表(got)信息
.got.pltPROGBITS用于存储文件的程序链接表(plt)信息。程序链接表是一个数据结构,用于描述程序中的程序链接(例如,库和其他程序中的函数和变量)。
.dataPROGBITS初始化变量的默认值
.bssPROGBITS为未初始化的变量保留空间
.commentPROGBITS用于存储文件的注释信息。注释信息是一个字符串,用于描述文件的注释信息。
.shstrtabSTRTAB以NULL 结尾的字符串数组,包含所有的节的名称,通过节名称进行索引
.symtabSYMTAB包含一个符合表,该表是一个ELF65_Sym 结构体数组,每个条目都将符合名与二进制文件中的代码和数据相关联,包含符号名的实际字符串保存到strtab 中
.strtabSTRTAB符号名的实际字符串

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/846730.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

一文了解经典报童模型的扩展问题

文章目录 1 引言2 经典报童模型3 综述文章4 模型扩展4.1 扩展目标函数4.2 增加约束条件4.3 增加优化变量4.4 扩展模型参数4.5 扩展问题场景 5 总结6 相关阅读 1 引言 时间过的真快呀,已经6月份了。距离上一篇文章发表,已经过去了将近一个月,…

godot.bk:how to add map to the game

1.项目构建如下,map是我们点击start之后才渲染出来的 mian.tscn --main.gd --background(textureact) --start(button) --button.gd sourceFile map.tscn --tilemap --tileset 2.main.gd:注意main.gd并不定义信号,它只是接收信号而已 extend…

Python冷知识

Python作为一种广泛使用的编程语言,有许多功能和特性可能不为初学者或普通用户所熟知。以下是一些相对冷门但有趣的Python知识: 魔术方法:Python中有一些特殊的方法,通常以双下划线__开头和结尾,被称为魔术方法(或特殊方法)。例如,__init__用于初始化对象,__str__返回…

冒泡排序与快速排序

博主主页: 码农派大星. 数据结构专栏:Java数据结构 数据库专栏:MySQL数据库 关注博主带你了解更多数据结构知识 1.冒泡排序 冒泡排序 private static void swap(int[] arrary,int i,int j){int tmp arrary[i];arrary[i] arrary[j];arrary[j] tmp;public static void bubbl…

Java筑基-面向对象

Java-面向对象 一、类和对象1、类和对象的关系2、创建类3、创建对象4、成员变量与局部变量5、构造器5.1、创建对象的过程5.2、构造器的格式5.3、构造器和方法的区别5.4、构造器的作用5.5、构造器的重载 6、this关键字用法:6.1、this可以修饰属性6.2、this可以修饰方…

【Ubuntu常用命令】终端个人常用命令总结

【Ubuntu常用命令】终端常用命令总结 查看硬盘挂载情况查看内存占用情况移动或重命名文件和目录复制文件或目录conda安装本地文件 查看硬盘挂载情况 mount 命令会列出当前系统上所有已挂载的文件系统。它会显示挂载点、文件系统类型、挂载选项等信息 mount df 命令用于显示文…

使用 pm2 或 screen 等工具来管理和后台运行你的 Node.js 应用

使用 pm2 或 screen 等工具来管理和后台运行你的 Node.js 应用。 使用 pm2 pm2 是一个用于 Node.js 应用的进程管理工具,提供了守护进程、日志管理和应用重启等功能。 安装 pm2: npm install pm2 -g启动你的 Node.js 应用: pm2 start se…

outlook邮箱使用技巧

Microsoft Outlook 是一款广泛使用的电子邮件客户端,它提供了许多功能来帮助用户有效地管理电子邮件、日程、联系人等。下面是一些实用的Outlook邮箱使用技巧,可以帮助你提高工作效率和邮件管理效能: ### 1. 快速搜索 - **使用搜索功能**&am…

【一小时学会Charles抓包详细教程】Charles 抓包相关设置 (7)

🚀 个人主页 极客小俊 ✍🏻 作者简介:程序猿、设计师、技术分享 🐋 希望大家多多支持, 我们一起学习和进步! 🏅 欢迎评论 ❤️点赞💬评论 📂收藏 📂加关注 Charles 抓包相…

数据库(18)——DCL权限控制

MySQL常用权限 权限说明ALL,ALL PRIVILEGES所有权限SELECT查询数据INSERT插入数据UPDATE修改数据DELETE删除数据ALTER修改表DROP删除数据库/表/视图CREATE创建数据库/表 DCL语法 查询权限 SHOW GRANTS FOR 用户名主机名; 查询hello的权限 SHOW GRANTS FOR hellolocalhost; 授…

02.了解容器的发展史

容器技术的发展过程 1):chroot技术,新建一个子系统(拥有自己完整的系统文件) 参考资料:https://www.ibm.com/developerworks/cn/linux/l-cn-chroot/ chang root 使用chroot监狱限制SSH用户访问指定目录和使用指定命…

【ARIMA时序预测】基于支持向量机结合ARIMA-SVM实现风电功率预测附matlab代码

% 步骤1:加载风电功率数据 load(‘wind_power_data.mat’); % 假设数据存储在变量power_data中 % 步骤2:划分训练集和测试集 trainRatio 0.7; % 训练集比例 trainSize floor(trainRatio * length(power_data)); trainData power_data(1:trainSize); …

GCB | 基于36年5个生态系统观测数据发现表层土壤深度提高生态系统的生产力和稳定性

陆地生态系统生产力对全球粮食安全和促进碳固存至关重要,但生产力受到气候变化以及火灾、干旱、洪水、霜冻频率增加和生物多样性减少的压力。了解控制生态系统初级生产力变异的不同因素和机制,为维持生态系统初级生产力和增强生态系统恢复力提供了科学依…

LeetCode题练习与总结:不同的子序列--115

一、题目描述 给你两个字符串 s 和 t ,统计并返回在 s 的 子序列 中 t 出现的个数,结果需要对 10^9 7 取模。 示例 1: 输入:s "rabbbit", t "rabbit" 输出:3 解释: 如下所示, 有 …

OJ1230进制的转换

答案&#xff1a; #include <bits/stdc.h> using namespace std; using lllong long; const int N10; int a[10]; char ch[]{0,1,2,3,4,5,6,7,8,9,A,B,C,D,E,F}; void solve() {int n,m;cin>>n>>m;string str;cin>>str;for(int i0;i<str.size();i)…

字符串操作java

题目&#xff1a; 描述 给定长度为n的只有小写字母的字符串s&#xff0c;进行m次操作&#xff0c;每次将[l,r]范围内所有c1字符改成c2&#xff0c;输出操作完的字符串 输入描述&#xff1a; 第一行两个数n,m 第二行一个字符串s 之后m行&#xff0c;每行两个数l 、r两个字符…

Android 系统APP提高CPU优先级,抢占不到CPU

系统APP提高CPU优先级,抢占不到CPU 1.在Android系统中,每个进程都有一个优先级,系统会根据进程的优先级来分配CPU资源。有时候android设备开机的时候负载过高,优先级较低抢占不到cpu,我们需要提高某个进程的优先级,以确保该进程能够顺利执行。 2.开发一个后台任务,需要在…

debian 将pdf 转换为jpg图片

打开终端并执行以下命令来安装这些工具&#xff1a; sudo apt-get update sudo apt-get install imagemagick ghostscript安装完成后&#xff0c;您可以使用convert命令转换PDF到JPEG&#xff1a; convert -density 300 input.pdf[0] output.jpg这里的-density 300参数是设置…

【STL】优先级队列用法介绍及其实现

目录 前言 用法介绍 模拟实现 仿函数 迭代器区间构造 插入 删除 队顶 判空 大小 完整代码 前言 优先级队列&#xff08;priority_queue&#xff09;是一种特殊的队列数据结构&#xff0c;其中每个元素都有一个“优先级”。优先级最高的元素最先出队&#xff0c;而优…

内网域中NTLM中继那些事儿

0x01 初识NTLM协议 基本概念&#xff1a;NTLM(NT LAN Manager)认证是一种早期的Windows网络身份认证协议。它在Windows系统中用于验证用户的身份&#xff0c;并提供对网络资源的访问控制&#xff0c;它是一种基于Challenge/Response的认证机制。 认证流程 NTLM协议Challenge…