typedef struct {
  unsigned char  e_ident[EI_NIDENT];  /* File identification. */
  Elf32_Half  e_type;    /* File type. */
  Elf32_Half  e_machine;  /* Machine architecture. */
  Elf32_Word  e_version;  /* ELF format version. */
  Elf32_Addr  e_entry;  /* Entry point. */
  Elf32_Off  e_phoff;  /* Program header file offset. */
  Elf32_Off  e_shoff;  /* Section header file offset. */
  Elf32_Word  e_flags;  /* Architecture-specific flags. */
  Elf32_Half  e_ehsize;  /* Size of ELF header in bytes. */
  Elf32_Half  e_phentsize;  /* Size of program header entry. */
  Elf32_Half  e_phnum;  /* Number of program header entries. */
  Elf32_Half  e_shentsize;  /* Size of section header entry. */
  Elf32_Half  e_shnum;  /* Number of section header entries. */
  Elf32_Half  e_shstrndx;  /* Section name strings section. */
} Elf32_Ehdr;

010 打开之后是这样的:

ELF header ELF 文件头 里面是基础信息,装载时看的是 Program header,涉及到 ELF header 里的变量 e_phoff 偏移 e_phentsize 每个的大小和 e_phnum 数量。同样,链接时看的是 Section header ,涉及到 ELF header 里的变量 e_shoff 偏移,e_shentsize 每个的大小和 e_shnum 数量,e_shstrndx section 的字符串表

Program header table 一般紧贴着 ELF header,里面长这样

类型、权限、文件里的位置、要加载到虚拟内存中的位置、(现在没用的物理位置)、在文件中的大小、在内存中的大小、对齐大小

Section header 里面长这样 段表

名字、类型、权限、虚拟内存位置(addr)、对应的文件偏移(offset)、大小、关联的 section、其他信息、对齐、某些特别的段的表的每一项的大小(如重定位表、符号表、哈希表、动态链接表这些)

s_name64_t s_name;               /* Section name */
s_type64_e s_type;               /* Section type */
s_flags64_e s_flags;             /* Section attributes */
Elf64_Addr s_addr;               /* Virtual address in memory */
Elf64_Off s_offset <format=hex>; /* Offset in file */
Elf64_Xword s_size;              /* Size of section */
Elf64_Word s_link;               /* Link to other section */
Elf64_Word s_info;               /* Miscellaneous information */
Elf64_Xword s_addralign;         /* Address alignment boundary */
Elf64_Xword s_entsize;           /* Entry size, if section has table */

dynamic 段里长这样:

linker 需要获取里面的比较关键的信息是:(d_tag

  • DT_NEEDED :依赖的共享库列表。
  • DT_SYMTAB :动态符号表( .dynsym )地址。
  • DT_RELA / DT_REL :重定位表( .rela.plt )地址。
  • DT_JMPREL :PLT 重定位表(通常与 .rela.plt 相同)。
  • DT_HASH / DT_GNU_HASH :符号哈希表地址。

动态链接的步骤和实现