/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│ │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│ ╞══════════════════════════════════════════════════════════════════════════════╡ │ Copyright 2020 Justine Alexandra Roberts Tunney │ │ │ │ This program is free software; you can redistribute it and/or modify │ │ it under the terms of the GNU General Public License as published by │ │ the Free Software Foundation; version 2 of the License. │ │ │ │ This program is distributed in the hope that it will be useful, but │ │ WITHOUT ANY WARRANTY; without even the implied warranty of │ │ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU │ │ General Public License for more details. │ │ │ │ You should have received a copy of the GNU General Public License │ │ along with this program; if not, write to the Free Software │ │ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA │ │ 02110-1301 USA │ ╚─────────────────────────────────────────────────────────────────────────────*/ #include "libc/bits/safemacros.h" #include "libc/calls/calls.h" #include "libc/calls/struct/stat.h" #include "libc/conv/conv.h" #include "libc/elf/elf.h" #include "libc/elf/struct/rela.h" #include "libc/elf/struct/shdr.h" #include "libc/errno.h" #include "libc/log/check.h" #include "libc/log/log.h" #include "libc/stdio/stdio.h" #include "libc/str/str.h" #include "libc/sysv/consts/map.h" #include "libc/sysv/consts/o.h" #include "libc/sysv/consts/prot.h" #include "tool/decode/lib/asmcodegen.h" #include "tool/decode/lib/elfidnames.h" #include "tool/decode/lib/flagger.h" #include "tool/decode/lib/titlegen.h" /** * @fileoverview ELF executable metadata disassembler. */ static const char *path; static struct stat st[1]; static Elf64_Ehdr *elf; static void startfile(void) { showtitle("αcτµαlly pδrταblε εxεcµταblε", "tool/decode/elf", basename(path), NULL, &kModelineAsm); printf("#include \"libc/elf.h\"\n\n", path); } static void printelfehdr(void) { show(".ascii", format(b1, "%`'.*s", 4, (const char *)&elf->e_ident[0]), "magic"); show(".byte", firstnonnull(findnamebyid(&kElfClassNames[0], elf->e_ident[EI_CLASS]), format(b1, "%d", elf->e_ident[EI_CLASS])), "elf->e_ident[EI_CLASS]"); show(".byte", firstnonnull(findnamebyid(kElfDataNames, elf->e_ident[EI_DATA]), format(b1, "%d", elf->e_ident[EI_DATA])), "elf->e_ident[EI_DATA]"); show(".byte", format(b1, "%d", elf->e_ident[EI_VERSION]), "elf->e_ident[EI_VERSION]"); show(".byte", firstnonnull(findnamebyid(kElfOsabiNames, elf->e_ident[EI_OSABI]), format(b1, "%d", elf->e_ident[EI_OSABI])), "elf->e_ident[EI_OSABI]"); show(".byte", format(b1, "%d", elf->e_ident[EI_ABIVERSION]), "elf->e_ident[EI_ABIVERSION]"); show(".byte", format(b1, "%d,%d,%d,%d,%d,%d,%d", elf->e_ident[EI_PAD + 0], elf->e_ident[EI_PAD + 1], elf->e_ident[EI_PAD + 2], elf->e_ident[EI_PAD + 3], elf->e_ident[EI_PAD + 4], elf->e_ident[EI_PAD + 5], elf->e_ident[EI_PAD + 6]), "padding"); show(".org", "0x10", NULL); show(".short", firstnonnull(findnamebyid(kElfTypeNames, elf->e_type), format(b1, "%hd", elf->e_type)), "elf->e_type"); show(".short", firstnonnull(findnamebyid(kElfMachineNames, elf->e_machine), format(b1, "%hd", elf->e_machine)), "elf->e_machine"); show(".long", format(b1, "%d", elf->e_version), "elf->e_version"); show(".quad", format(b1, "%#x", elf->e_entry), "elf->e_entry"); show(".quad", format(b1, "%#x", elf->e_phoff), "elf->e_phoff"); show(".quad", format(b1, "%#x", elf->e_shoff), "elf->e_shoff"); show(".long", format(b1, "%#x", elf->e_flags), "elf->e_flags"); show(".short", format(b1, "%hd", elf->e_ehsize), "elf->e_ehsize"); show(".short", format(b1, "%hd", elf->e_phentsize), "elf->e_phentsize"); show(".short", format(b1, "%hd", elf->e_phnum), "elf->e_phnum"); show(".short", format(b1, "%hd", elf->e_shentsize), "elf->e_shentsize"); show(".short", format(b1, "%hd", elf->e_shnum), "elf->e_shnum"); show(".short", format(b1, "%hd", elf->e_shstrndx), "elf->e_shstrndx"); } static void printelfsegmentheader(int i) { Elf64_Phdr *phdr = getelfsegmentheaderaddress(elf, st->st_size, i); printf("/\tElf64_Phdr *phdr = getelfsegmentheaderaddress(elf, st->st_size, " "%d)\n", i); printf(".Lph%d:", i); show(".long", firstnonnull(findnamebyid(kElfSegmentTypeNames, phdr->p_type), format(b1, "%#x", phdr->p_type)), "phdr->p_type"); show(".long", RecreateFlags(kElfSegmentFlagNames, phdr->p_flags), "phdr->p_flags"); show(".quad", format(b1, "%#x", phdr->p_offset), "phdr->p_offset"); show(".quad", format(b1, "%#x", phdr->p_vaddr), "phdr->p_vaddr"); show(".quad", format(b1, "%#x", phdr->p_paddr), "phdr->p_paddr"); show(".quad", format(b1, "%#x", phdr->p_filesz), "phdr->p_filesz"); show(".quad", format(b1, "%#x", phdr->p_memsz), "phdr->p_memsz"); show(".quad", format(b1, "%#x", phdr->p_align), "phdr->p_align"); fflush(stdout); } static void printelfsegmentheaders(void) { printf("\n"); printf("\t.org\t%#x\n", elf->e_phoff); for (unsigned i = 0; i < elf->e_phnum; ++i) printelfsegmentheader(i); } static void printelfsectionheader(int i, char *shstrtab) { Elf64_Shdr *shdr; shdr = getelfsectionheaderaddress(elf, st->st_size, i); printf("/\tElf64_Shdr *shdr = getelfsectionheaderaddress(elf, st->st_size, " "%d)\n", i); printf(".Lsh%d:", i); show(".long", format(b1, "%d", shdr->sh_name), format(b2, "%`'s == getelfstring(elf, st->st_size, shstrtab, shdr->sh_name)", getelfstring(elf, st->st_size, shstrtab, shdr->sh_name))); show(".long", firstnonnull(findnamebyid(kElfSectionTypeNames, shdr->sh_type), format(b1, "%d", shdr->sh_type)), "shdr->sh_type"); show(".long", RecreateFlags(kElfSectionFlagNames, shdr->sh_flags), "shdr->sh_flags"); show(".quad", format(b1, "%#x", shdr->sh_addr), "shdr->sh_addr"); show(".quad", format(b1, "%#x", shdr->sh_offset), "shdr->sh_offset"); show(".quad", format(b1, "%#x", shdr->sh_size), "shdr->sh_size"); show(".long", format(b1, "%#x", shdr->sh_link), "shdr->sh_link"); show(".long", format(b1, "%#x", shdr->sh_info), "shdr->sh_info"); show(".quad", format(b1, "%#x", shdr->sh_addralign), "shdr->sh_addralign"); show(".quad", format(b1, "%#x", shdr->sh_entsize), "shdr->sh_entsize"); fflush(stdout); } static void printelfsectionheaders(void) { Elf64_Half i; char *shstrtab = getelfsectionnamestringtable(elf, st->st_size); if (shstrtab) { printf("\n"); printf("\t.org\t%#x\n", elf->e_shoff); for (i = 0; i < elf->e_shnum; ++i) { printelfsectionheader(i, shstrtab); } printf("\n/\t%s\n", "elf->e_shstrndx"); printf("\t.org\t%#x\n", getelfsectionheaderaddress(elf, st->st_size, elf->e_shstrndx) ->sh_offset); for (i = 0; i < elf->e_shnum; ++i) { Elf64_Shdr *shdr = getelfsectionheaderaddress(elf, st->st_size, i); const char *str = getelfstring(elf, st->st_size, shstrtab, shdr->sh_name); show(".asciz", format(b1, "%`'s", str), NULL); } } } static void printelfsymbolinfo(Elf64_Sym *sym) { int bind = (sym->st_info >> 4) & 0xf; const char *bindname = findnamebyid(kElfSymbolBindNames, bind); int type = (sym->st_info >> 0) & 0xf; const char *typename = findnamebyid(kElfSymbolTypeNames, type); show(".byte", format(b1, "%s%s%s", bindname ? format(b2, "%s<<4", bindname) : "", bindname && typename ? "|" : "", firstnonnull(typename, "")), "sym->st_info"); } static void printelfsymbolother(Elf64_Sym *sym) { int visibility = sym->st_other & 0x3; const char *visibilityname = findnamebyid(kElfSymbolVisibilityNames, visibility); int other = sym->st_other & ~0x3; show(".byte", format(b1, "%s%s%s", firstnonnull(visibilityname, ""), other && visibilityname ? "+" : "", other ? format(b2, "%d", other) : ""), "sym->st_other"); } static void printelfsymbol(Elf64_Sym *sym, char *strtab, char *shstrtab) { show(".long", format(b1, "%d", sym->st_name), format(b2, "%`'s (sym->st_name)", getelfstring(elf, st->st_size, strtab, sym->st_name))); printelfsymbolinfo(sym); printelfsymbolother(sym); show(".short", format(b1, "%d", sym->st_shndx), format(b2, "%s sym->st_shndx", sym->st_shndx < 0xff00 ? format(b1, "%`'s", getelfstring(elf, st->st_size, shstrtab, getelfsectionheaderaddress( elf, st->st_size, sym->st_shndx) ->sh_name)) : findnamebyid(kElfSpecialSectionNames, sym->st_shndx))); show(".quad", format(b1, "%#x", sym->st_value), "sym->st_value"); show(".quad", format(b1, "%#x", sym->st_size), "sym->st_size"); } static void printelfsymboltable(void) { size_t i, symcount = 0; Elf64_Sym *symtab = getelfsymboltable(elf, st->st_size, &symcount); char *strtab = getelfstringtable(elf, st->st_size); char *shstrtab = getelfsectionnamestringtable(elf, st->st_size); if (symtab && strtab) { printf("\n\n"); printf("\t.org\t%#x\n", (intptr_t)symtab - (intptr_t)elf); for (i = 0; i < symcount; ++i) { printf(".Lsym%d:\n", i); printelfsymbol(&symtab[i], strtab, shstrtab); } } } static char *getelfsymbolname(const Elf64_Ehdr *elf, size_t mapsize, const char *strtab, const char *shstrtab, const Elf64_Sym *sym) { char *res; const Elf64_Shdr *shdr; if (elf && sym && ((shstrtab && !sym->st_name && ELF64_ST_TYPE(sym->st_info) == STT_SECTION && (shdr = getelfsectionheaderaddress(elf, mapsize, sym->st_shndx)) && (res = getelfstring(elf, mapsize, shstrtab, shdr->sh_name))) || (strtab && (res = getelfstring(elf, mapsize, strtab, sym->st_name))))) { return res; } else { return NULL; } } static void printelfrelocations(void) { int sym; size_t i, j; const Elf64_Sym *syms; const Elf64_Rela *rela; const Elf64_Shdr *shdr, *boop; char *strtab, *shstrtab, *symbolname; strtab = getelfstringtable(elf, st->st_size); shstrtab = getelfsectionnamestringtable(elf, st->st_size); for (i = 0; i < elf->e_shnum; ++i) { if ((shdr = getelfsectionheaderaddress(elf, st->st_size, i)) && shdr->sh_type == SHT_RELA && (rela = getelfsectionaddress(elf, st->st_size, shdr))) { printf("\n/\t%s\n", getelfsectionname(elf, st->st_size, shdr)); printf("\t.org\t%#x\n", (intptr_t)rela - (intptr_t)elf); for (j = 0; ((uintptr_t)rela + sizeof(Elf64_Rela) <= min((uintptr_t)elf + st->st_size, (uintptr_t)elf + shdr->sh_offset + shdr->sh_size)); ++rela, ++j) { boop = getelfsectionheaderaddress(elf, st->st_size, shdr->sh_link); syms = getelfsectionaddress(elf, st->st_size, boop); sym = ELF64_R_SYM(rela->r_info); symbolname = getelfsymbolname(elf, st->st_size, strtab, shstrtab, &syms[sym]); printf("/\t%s+%#lx → %s%c%#lx\n", getelfstring( elf, st->st_size, shstrtab, getelfsectionheaderaddress(elf, st->st_size, shdr->sh_info) ->sh_name), rela->r_offset, symbolname, rela->r_addend >= 0 ? '+' : '-', abs(rela->r_addend)); printf("%s_%zu_%zu:\n", ".Lrela", i, j); show(".quad", format(b1, "%#lx", rela->r_offset), "rela->r_offset"); show(".long", format(b1, "%s%s", "R_X86_64_", findnamebyid(kElfNexgen32eRelocationNames, ELF64_R_TYPE(rela->r_info))), "ELF64_R_TYPE(rela->r_info)"); show(".long", format(b1, "%d", ELF64_R_SYM(rela->r_info)), "ELF64_R_SYM(rela->r_info)"); show(".quad", format(b1, "%#lx", rela->r_addend), "rela->r_addend"); } } } } int main(int argc, char *argv[]) { showcrashreports(); if (argc != 2) { fprintf(stderr, "usage: %`s FILE: %s\n", argv[0]); return 1; } path = argv[1]; int64_t fd = open(path, O_RDONLY); if (fd == -1) { if (errno == ENOENT) { fprintf(stderr, "error: %`s not found\n", path); exit(1); } perror("open"); exit(1); } fstat(fd, st); CHECK_NE(MAP_FAILED, (elf = mmap(NULL, st->st_size, PROT_READ, MAP_SHARED, fd, 0))); if (memcmp(elf->e_ident, ELFMAG, 4) != 0) { fprintf(stderr, "error: not an elf executable: %'s\n", path); exit(1); } startfile(); printelfehdr(); printelfsegmentheaders(); printelfsectionheaders(); printelfrelocations(); printelfsymboltable(); munmap(elf, st->st_size); close(fd); return 0; }