ELF文件解析二
ELF header
ELF header的定义可以在 /usr/include/elf.h 中找到。Elf32_Ehdr是32位 ELF header的结构体。Elf64_Ehdr是64位ELF header的结构体。
1 | typedef struct |
64位和32位只是个别字段长度不同,比如 Elf64_Addr 和 Elf64_Off 都是64位无符号整数。而Elf32_Addr 和 Elf32_Off是32位无符号整数。这导致ELF header的所占的字节数不同。32位的ELF header占52个字节,64位的ELF header占64个字节.
ELF header 详解
- e_ident占16个字节。前四个字节被称作ELF的Magic Number。后面的字节描述了ELF文件内容如何解码等信息。
- e_type,2字节,描述了ELF文件的类型。以下取值有意义:从ET_LOPROC到 ET_HIPROC 的值,包含特定于处理器的语义。
1
2
3
4
5
6
7
8ET_NONE, 0, No file type
ET_REL, 1, Relocatable file(可重定位文件,通常是文件名以.o结尾,目标文件)
ET_EXEC, 2, Executable file (可执行文件)
ET_DYN, 3, Shared object file (动态库文件))
ET_CORE, 4, Core file (core文件)
ET_NUM, 5,(表示已经定义了5种文件类型)
ET_LOPROC, 0xff00, Processor-specific
ET_HIPROC, 0xffff, Processor-specific - e_machine,2字节。描述了文件面向的架构,可取值如下(因为文档较老,现在有更多取值,参见/usr/include/elf.h中的EM_开头的宏定义):
1
2
3
4
5
6
7
8
9EM_NONE, 0, No machine
EM_M32, 1, AT&T WE 32100
EM_SPARC, 2, SPARC
EM_386, 3, Intel 80386
EM_68K, 4, Motorola 68000
EM_88K, 5, Motorola 88000
EM_860, 7, Intel 80860
EM_MIPS, 8, MIPS RS3000
... ... - e_version,2字节,描述了ELF文件的版本号,合法取值如下:
1
2
3EV_NONE, 0, Invalid version
EV_CURRENT, 1, Current version,
EV_NUM, 2, (表示已经定义了2种版本号) - e_entry,(32位4字节,64位8字节),执行入口点,如果文件没有入口点,这个域保持0。
- e_phoff, (32位4字节,64位8字节),program header table的offset,如果文件没有PH,这个值是0。
- e_shoff, (32位4字节,64位8字节), section header table 的offset,如果文件没有SH,这个值是0。
- e_flags, 4字节,特定于处理器的标志,32位和64位Intel架构都没有定义标志,因此eflags的值是0。
- e_ehsize, 2字节,ELF header的大小,32位ELF是52字节,64位是64字节。
- e_phentsize,2字节。program header table中每个入口的大小。
- e_phnum, 2字节。如果文件没有program header table, e_phnum的值为0。e_phentsize乘以e_phnum就得到了整个program header table的大小。
- e_shentsize, 2字节,section header table中entry的大小,即每个section header占多少字节。
- e_shnum, 2字节,section header table中header的数目。如果文件没有section header table, e_shnum的值为0。e_shentsize乘以e_shnum,就得到了整个section header table的大小。
- e_shstrndx, 2字节。section header string table index. 包含了section header table中section name string table。如果没有section name string table, e_shstrndx的值是SHN_UNDEF.
e_ident
如图,前4个字节是ELF的Magic Number,固定为7f 45 4c 46。
第5个字节指明ELF文件是32位还是64位的。
第6个字节指明了数据的编码方式,即我们通常说的little endian或是big endian。little endian
第7个字节指明了ELF header的版本号,目前值都是1。
第8-16个字节,都填充为0。
readelf读取ELF header
我们使用readelf -h
比如我本地有执行文件hello,我执行readelf -h hello,结果例如:
1 | ELF Header: |