微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

ELF文件TLS和LOAD程序部分

int i; int main() { return i; }

readelf -l编译readelf -l显示来自elf的程序头文件

Elf file type is EXEC (Executable file) Entry point 0xxxxx30 There are 6 program headers,starting at offset 52 Program Headers: Type Offset VirtAddr PhysAddr FileSiz MemSiz Flg Align LOAD 0x000000 0x08048000 0x08048000 0x79868 0x79868 RE 0x1000 > LOAD 0x079f94 0x080c2f94 0x080c2f94 0x0078c 0x02254 RW 0x1000 << NOTE 0x0000f4 0x080480f4 0x080480f4 0x00020 0x00020 R 0x4 > TLS 0x079f94 0x080c2f94 0x080c2f94 0x00010 0x0002c R 0x4 << GNU_STACK 0x000000 0x00000000 0x00000000 0x00000 0x00000 RW 0x4 pax_FLAGS 0x000000 0x00000000 0x00000000 0x00000 0x00000 0x4 Section to Segment mapping: Segment Sections... 00 .note.ABI-tag .init .text __libc_freeres_fn .fini .rodata __libc_subfreeres __libc_atexit .eh_frame .gcc_except_table 01 .tdata .ctors .dtors .jcr .data.rel.ro .got .got.plt .data .bss __libc_freeres_ptrs 02 .note.ABI-tag 03 .tdata .tbss

有人可以解释,为什么第二个和第四个程序头相交(它们以相同的偏移量0x079f94和VirtAddr 0x080c2f94开始)。

此外,段部分.tdata被引用两次。

如何为第一个线程(程序本身)加载PT_TLS和PT_LOAD ? .tbss在什么地方记忆?

在ELF二进制中导入名称

ELF文件的.bss部分应该放在哪里?

如何防止用户读取存储在二进制文件中的string?

创buildELF二进制文件,而不使用libelf或其他库

VMA和ELF分部之间的关系

为什么我的可执行文件中的入口点地址是0x8048330(0x330是.text段的偏移量)

用于修改ELF二进制文件的dynamic部分的工具

强制GNU链接生成32位ELF可执行文件

为什么Linux / GNU链接器select地址0x400000?

GCC编译的二进制文件W /不同的大小?

首先.tdata部分 – 是TLS数据的“初始映像”。 这是TLS变量的初始值,它将在每个线程中使用(也在主线程中)。 在crt (我认为)有一个TLS的初始图像复制到主线程的TLS。 相同的代码在pthread_create 。

没有加载PT_TLS,因为PT_LOAD和PT_LOAD已经包含这个PT_TLS。 我认为PT_TLS是用于初始图像 – 因为它比整个线程本地数据(tbss + tdata> size(PT_TLS))短。

TLS代表“线程局部存储”。

为了允许将在编译时分配的数据与单独的执行线程相关联,可以使用线程本地存储部分来指定这些数据的大小和初始内容。 实现不需要支持线程本地存储。 PT_TLS程序入口具有以下成员:

Member Value p_offset File offset of the TLS initialization image p_vaddr Virtual memory address of the TLS initialization image p_paddr reserved p_filesz Size of the TLS initialization image p_memsz Total size of the TLS template p_flags PF_R p_align Alignment of the TLS template

TLS模板由具有标志SHF_TLS的所有部分的组合形成。 保存初始化数据的TLS模板部分是TLS初始化图像。 (TLS模板的其余部分是SHT_NOBITS类型的一个或多个部分。)

就映射内存区而言,我认为内核只查看PT_LOAD段并将它们映射到它们。 (内核也查看PT_GNU_STACK来判断堆栈是否应该与Execute权限进行映射。)查看binfmt_elf.c:load_elf_binary()以获取相关代码

通过libc读取PT_TLS段以找出要设置线程本地存储器的内存。 查看相关代码的__libc_setup_tls()。

PT_TLS段与PT_LOAD段交叉,以便将其映射到进程内存中。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 [email protected] 举报,一经查实,本站将立刻删除。

相关推荐