elf - 如何理解elf中Program Headers中Offset和VirAddr的区别?
问题描述
有一个共享库elf文件,我readelf -l
用来查看程序头文件,输出为:
Elf file type is DYN (Shared object file)
Entry point 0x0
There are 11 program headers, starting at offset 52
Program Headers:
Type Offset VirtAddr PhysAddr FileSiz MemSiz Flg Align
PHDR 0x000034 0x00000034 0x00000034 0x00100 0x00100 R 0x4
INTERP 0x000194 0x00000194 0x00000194 0x00013 0x00013 R 0x1
[Requesting program interpreter: /system/bin/linker]
LOAD 0x000000 0x00000000 0x00000000 0x3aa8c4 0x3aa8c4 R E 0x1000
LOAD 0x3ab1cc 0x003ac1cc 0x003ac1cc 0x062c0 0x25ee4 RW 0x1000
LOAD 0x3b2000 0x003d3000 0x003d3000 0x02561 0x02561 R E 0x1000
LOAD 0x3b4e8c 0x003d6e8c 0x003d6e8c 0x00298 0x00299 RW 0x1000
LOAD 0x3b5268 0x003d8268 0x003d8268 0x00128 0x00128 RW 0x1000
DYNAMIC 0x3b5268 0x003d8268 0x003d8268 0x00128 0x00128 RW 0x4
GNU_STACK 0x000000 0x00000000 0x00000000 0x00000 0x00000 RW 0
EXIDX 0x2e71e8 0x002e71e8 0x002e71e8 0x0b558 0x0b558 R 0x4
GNU_RELRO 0x3ab1cc 0x003ac1cc 0x003ac1cc 0x01e34 0x01e34 RW 0x4
Section to Segment mapping:
Segment Sections...
00
01 .interp
02 .interp .dynsym .dynstr .hash .gnu.version .gnu.version_d .rel.dyn .plt .text .ARM.extab .ARM.exidx .rodata
03 .data.rel.ro.local .fini_array .data.rel.ro .got .data .bss
04 .rel.plt
05 .init_array
06 .dynamic
07 .dynamic
08
09 .ARM.exidx
10 .data.rel.ro.local .fini_array .data.rel.ro .got
如果以下结构表示程序头:
typedef struct {
uint32_t p_type;
Elf32_Off p_offset;
Elf32_Addr p_vaddr;
Elf32_Addr p_paddr;
uint32_t p_filesz;
uint32_t p_memsz;
uint32_t p_flags;
uint32_t p_align;
} Elf32_Phdr;
那么我的问题是:如何理解与输出对应的p_offset
和之间的区别?他们会永远一样吗?并且它们会被动态加载的过程改变吗?p_vaddr
Offset
VirtAddr
readelf -l
解决方案
如何理解 readelf -l 输出中对应 Offset 和 VirtAddr 的 p_offset 和 p_vaddr 之间的区别?
运行时加载器将在虚拟地址(类似地向下舍入;该地址实际上会为对象添加一些大的多页偏移)处mmap
的偏移量.p_offset
(向下舍入到页面大小)处的一组页面。.p_vaddr
ET_DYN
他们会永远一样吗?
即使在您的示例中,它们也不相同:
LOAD 0x3ab1cc 0x003ac1cc
0x3ab1
!= 0x3ac1
。可以保证的是(.p_offset % pagesize == .p_vaddr % pagesize
否则mmap
将变得不可能)。
推荐阅读
- firebase - 为 Firebase 应用程序授权应用引擎版本
- javascript - 使用jquery每个函数循环项目并将项目包装到div容器中
- flutter - 有没有办法从飞镖/颤振中的音频中分割左右声道?
- c# - 自定义 IdentityDb 上下文以使用 ApplicationRole
- r - 如何在 ggplot 瀑布图上添加误差线?
- javascript - Redux Form 未使用初始值初始化
- bash - 将数组/列表从 bash 脚本传递到 ansible 循环
- python - 给定一个数据框,我如何检查列的值是否按递增顺序排列而没有任何缺失的数字?
- laravel - 显示订单的产品 Vuejs & Laravel
- php - 如何在php的csv文件中添加sql oracle查询的结果?