18fe960a1SDavid E. O'Brien /*- 21de7b4b8SPedro F. Giffuni * SPDX-License-Identifier: BSD-2-Clause-FreeBSD 31de7b4b8SPedro F. Giffuni * 40529c56bSDavid E. O'Brien * Copyright (c) 2003 David O'Brien. All rights reserved. 58fe960a1SDavid E. O'Brien * Copyright (c) 2001 Jake Burkholder 68fe960a1SDavid E. O'Brien * All rights reserved. 78fe960a1SDavid E. O'Brien * 88fe960a1SDavid E. O'Brien * Redistribution and use in source and binary forms, with or without 98fe960a1SDavid E. O'Brien * modification, are permitted provided that the following conditions 108fe960a1SDavid E. O'Brien * are met: 118fe960a1SDavid E. O'Brien * 1. Redistributions of source code must retain the above copyright 128fe960a1SDavid E. O'Brien * notice, this list of conditions and the following disclaimer. 138fe960a1SDavid E. O'Brien * 2. Redistributions in binary form must reproduce the above copyright 148fe960a1SDavid E. O'Brien * notice, this list of conditions and the following disclaimer in the 158fe960a1SDavid E. O'Brien * documentation and/or other materials provided with the distribution. 168fe960a1SDavid E. O'Brien * 178fe960a1SDavid E. O'Brien * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 188fe960a1SDavid E. O'Brien * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 198fe960a1SDavid E. O'Brien * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 208fe960a1SDavid E. O'Brien * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 218fe960a1SDavid E. O'Brien * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 228fe960a1SDavid E. O'Brien * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 238fe960a1SDavid E. O'Brien * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 248fe960a1SDavid E. O'Brien * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 258fe960a1SDavid E. O'Brien * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 268fe960a1SDavid E. O'Brien * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 278fe960a1SDavid E. O'Brien * SUCH DAMAGE. 288fe960a1SDavid E. O'Brien */ 298fe960a1SDavid E. O'Brien 308fe960a1SDavid E. O'Brien #include <sys/cdefs.h> 318fe960a1SDavid E. O'Brien __FBSDID("$FreeBSD$"); 328fe960a1SDavid E. O'Brien 338fe960a1SDavid E. O'Brien #include <sys/types.h> 349efc7e72SChristian Brueffer 359efc7e72SChristian Brueffer #include <sys/capsicum.h> 368fe960a1SDavid E. O'Brien #include <sys/elf32.h> 378fe960a1SDavid E. O'Brien #include <sys/elf64.h> 38ac2ded1dSMarcel Moolenaar #include <sys/endian.h> 398fe960a1SDavid E. O'Brien #include <sys/mman.h> 408fe960a1SDavid E. O'Brien #include <sys/stat.h> 41a4e3fc54SMariusz Zaborski #include <capsicum_helpers.h> 428fe960a1SDavid E. O'Brien #include <err.h> 439efc7e72SChristian Brueffer #include <errno.h> 448fe960a1SDavid E. O'Brien #include <fcntl.h> 4575ca4dcfSDavid E. O'Brien #include <inttypes.h> 468fe960a1SDavid E. O'Brien #include <stddef.h> 478fe960a1SDavid E. O'Brien #include <stdio.h> 488fe960a1SDavid E. O'Brien #include <stdlib.h> 498fe960a1SDavid E. O'Brien #include <string.h> 508fe960a1SDavid E. O'Brien #include <unistd.h> 518fe960a1SDavid E. O'Brien 528fe960a1SDavid E. O'Brien #define ED_DYN (1<<0) 538fe960a1SDavid E. O'Brien #define ED_EHDR (1<<1) 548fe960a1SDavid E. O'Brien #define ED_GOT (1<<2) 558fe960a1SDavid E. O'Brien #define ED_HASH (1<<3) 568fe960a1SDavid E. O'Brien #define ED_INTERP (1<<4) 578fe960a1SDavid E. O'Brien #define ED_NOTE (1<<5) 588fe960a1SDavid E. O'Brien #define ED_PHDR (1<<6) 598fe960a1SDavid E. O'Brien #define ED_REL (1<<7) 608fe960a1SDavid E. O'Brien #define ED_SHDR (1<<8) 618fe960a1SDavid E. O'Brien #define ED_SYMTAB (1<<9) 628fe960a1SDavid E. O'Brien #define ED_ALL ((1<<10)-1) 63c35530f4SBrooks Davis #define ED_IS_ELF (1<<10) /* Exclusive with other flags */ 648fe960a1SDavid E. O'Brien 658fe960a1SDavid E. O'Brien #define elf_get_addr elf_get_quad 668fe960a1SDavid E. O'Brien #define elf_get_off elf_get_quad 678fe960a1SDavid E. O'Brien #define elf_get_size elf_get_quad 688fe960a1SDavid E. O'Brien 698fe960a1SDavid E. O'Brien enum elf_member { 708fe960a1SDavid E. O'Brien D_TAG = 1, D_PTR, D_VAL, 718fe960a1SDavid E. O'Brien 728fe960a1SDavid E. O'Brien E_CLASS, E_DATA, E_OSABI, E_TYPE, E_MACHINE, E_VERSION, E_ENTRY, 738fe960a1SDavid E. O'Brien E_PHOFF, E_SHOFF, E_FLAGS, E_EHSIZE, E_PHENTSIZE, E_PHNUM, E_SHENTSIZE, 748fe960a1SDavid E. O'Brien E_SHNUM, E_SHSTRNDX, 758fe960a1SDavid E. O'Brien 768fe960a1SDavid E. O'Brien N_NAMESZ, N_DESCSZ, N_TYPE, 778fe960a1SDavid E. O'Brien 788fe960a1SDavid E. O'Brien P_TYPE, P_OFFSET, P_VADDR, P_PADDR, P_FILESZ, P_MEMSZ, P_FLAGS, 798fe960a1SDavid E. O'Brien P_ALIGN, 808fe960a1SDavid E. O'Brien 818fe960a1SDavid E. O'Brien SH_NAME, SH_TYPE, SH_FLAGS, SH_ADDR, SH_OFFSET, SH_SIZE, SH_LINK, 828fe960a1SDavid E. O'Brien SH_INFO, SH_ADDRALIGN, SH_ENTSIZE, 838fe960a1SDavid E. O'Brien 848fe960a1SDavid E. O'Brien ST_NAME, ST_VALUE, ST_SIZE, ST_INFO, ST_SHNDX, 858fe960a1SDavid E. O'Brien 868fe960a1SDavid E. O'Brien R_OFFSET, R_INFO, 878fe960a1SDavid E. O'Brien 888fe960a1SDavid E. O'Brien RA_OFFSET, RA_INFO, RA_ADDEND 898fe960a1SDavid E. O'Brien }; 908fe960a1SDavid E. O'Brien 918fe960a1SDavid E. O'Brien typedef enum elf_member elf_member_t; 928fe960a1SDavid E. O'Brien 93bf70beceSEd Schouten static int elf32_offsets[] = { 948fe960a1SDavid E. O'Brien 0, 958fe960a1SDavid E. O'Brien 968fe960a1SDavid E. O'Brien offsetof(Elf32_Dyn, d_tag), offsetof(Elf32_Dyn, d_un.d_ptr), 978fe960a1SDavid E. O'Brien offsetof(Elf32_Dyn, d_un.d_val), 988fe960a1SDavid E. O'Brien 998fe960a1SDavid E. O'Brien offsetof(Elf32_Ehdr, e_ident[EI_CLASS]), 1008fe960a1SDavid E. O'Brien offsetof(Elf32_Ehdr, e_ident[EI_DATA]), 1018fe960a1SDavid E. O'Brien offsetof(Elf32_Ehdr, e_ident[EI_OSABI]), 1028fe960a1SDavid E. O'Brien offsetof(Elf32_Ehdr, e_type), offsetof(Elf32_Ehdr, e_machine), 1038fe960a1SDavid E. O'Brien offsetof(Elf32_Ehdr, e_version), offsetof(Elf32_Ehdr, e_entry), 1048fe960a1SDavid E. O'Brien offsetof(Elf32_Ehdr, e_phoff), offsetof(Elf32_Ehdr, e_shoff), 1058fe960a1SDavid E. O'Brien offsetof(Elf32_Ehdr, e_flags), offsetof(Elf32_Ehdr, e_ehsize), 1068fe960a1SDavid E. O'Brien offsetof(Elf32_Ehdr, e_phentsize), offsetof(Elf32_Ehdr, e_phnum), 1078fe960a1SDavid E. O'Brien offsetof(Elf32_Ehdr, e_shentsize), offsetof(Elf32_Ehdr, e_shnum), 1088fe960a1SDavid E. O'Brien offsetof(Elf32_Ehdr, e_shstrndx), 1098fe960a1SDavid E. O'Brien 1108fe960a1SDavid E. O'Brien offsetof(Elf_Note, n_namesz), offsetof(Elf_Note, n_descsz), 1118fe960a1SDavid E. O'Brien offsetof(Elf_Note, n_type), 1128fe960a1SDavid E. O'Brien 1138fe960a1SDavid E. O'Brien offsetof(Elf32_Phdr, p_type), offsetof(Elf32_Phdr, p_offset), 1148fe960a1SDavid E. O'Brien offsetof(Elf32_Phdr, p_vaddr), offsetof(Elf32_Phdr, p_paddr), 1158fe960a1SDavid E. O'Brien offsetof(Elf32_Phdr, p_filesz), offsetof(Elf32_Phdr, p_memsz), 1168fe960a1SDavid E. O'Brien offsetof(Elf32_Phdr, p_flags), offsetof(Elf32_Phdr, p_align), 1178fe960a1SDavid E. O'Brien 1188fe960a1SDavid E. O'Brien offsetof(Elf32_Shdr, sh_name), offsetof(Elf32_Shdr, sh_type), 1198fe960a1SDavid E. O'Brien offsetof(Elf32_Shdr, sh_flags), offsetof(Elf32_Shdr, sh_addr), 1208fe960a1SDavid E. O'Brien offsetof(Elf32_Shdr, sh_offset), offsetof(Elf32_Shdr, sh_size), 1218fe960a1SDavid E. O'Brien offsetof(Elf32_Shdr, sh_link), offsetof(Elf32_Shdr, sh_info), 1228fe960a1SDavid E. O'Brien offsetof(Elf32_Shdr, sh_addralign), offsetof(Elf32_Shdr, sh_entsize), 1238fe960a1SDavid E. O'Brien 1248fe960a1SDavid E. O'Brien offsetof(Elf32_Sym, st_name), offsetof(Elf32_Sym, st_value), 1258fe960a1SDavid E. O'Brien offsetof(Elf32_Sym, st_size), offsetof(Elf32_Sym, st_info), 1268fe960a1SDavid E. O'Brien offsetof(Elf32_Sym, st_shndx), 1278fe960a1SDavid E. O'Brien 1288fe960a1SDavid E. O'Brien offsetof(Elf32_Rel, r_offset), offsetof(Elf32_Rel, r_info), 1298fe960a1SDavid E. O'Brien 1308fe960a1SDavid E. O'Brien offsetof(Elf32_Rela, r_offset), offsetof(Elf32_Rela, r_info), 1318fe960a1SDavid E. O'Brien offsetof(Elf32_Rela, r_addend) 1328fe960a1SDavid E. O'Brien }; 1338fe960a1SDavid E. O'Brien 134bf70beceSEd Schouten static int elf64_offsets[] = { 1358fe960a1SDavid E. O'Brien 0, 1368fe960a1SDavid E. O'Brien 1378fe960a1SDavid E. O'Brien offsetof(Elf64_Dyn, d_tag), offsetof(Elf64_Dyn, d_un.d_ptr), 1388fe960a1SDavid E. O'Brien offsetof(Elf64_Dyn, d_un.d_val), 1398fe960a1SDavid E. O'Brien 1408fe960a1SDavid E. O'Brien offsetof(Elf32_Ehdr, e_ident[EI_CLASS]), 1418fe960a1SDavid E. O'Brien offsetof(Elf32_Ehdr, e_ident[EI_DATA]), 1428fe960a1SDavid E. O'Brien offsetof(Elf32_Ehdr, e_ident[EI_OSABI]), 1438fe960a1SDavid E. O'Brien offsetof(Elf64_Ehdr, e_type), offsetof(Elf64_Ehdr, e_machine), 1448fe960a1SDavid E. O'Brien offsetof(Elf64_Ehdr, e_version), offsetof(Elf64_Ehdr, e_entry), 1458fe960a1SDavid E. O'Brien offsetof(Elf64_Ehdr, e_phoff), offsetof(Elf64_Ehdr, e_shoff), 1468fe960a1SDavid E. O'Brien offsetof(Elf64_Ehdr, e_flags), offsetof(Elf64_Ehdr, e_ehsize), 1478fe960a1SDavid E. O'Brien offsetof(Elf64_Ehdr, e_phentsize), offsetof(Elf64_Ehdr, e_phnum), 1488fe960a1SDavid E. O'Brien offsetof(Elf64_Ehdr, e_shentsize), offsetof(Elf64_Ehdr, e_shnum), 1498fe960a1SDavid E. O'Brien offsetof(Elf64_Ehdr, e_shstrndx), 1508fe960a1SDavid E. O'Brien 1518fe960a1SDavid E. O'Brien offsetof(Elf_Note, n_namesz), offsetof(Elf_Note, n_descsz), 1528fe960a1SDavid E. O'Brien offsetof(Elf_Note, n_type), 1538fe960a1SDavid E. O'Brien 1548fe960a1SDavid E. O'Brien offsetof(Elf64_Phdr, p_type), offsetof(Elf64_Phdr, p_offset), 1558fe960a1SDavid E. O'Brien offsetof(Elf64_Phdr, p_vaddr), offsetof(Elf64_Phdr, p_paddr), 1568fe960a1SDavid E. O'Brien offsetof(Elf64_Phdr, p_filesz), offsetof(Elf64_Phdr, p_memsz), 1578fe960a1SDavid E. O'Brien offsetof(Elf64_Phdr, p_flags), offsetof(Elf64_Phdr, p_align), 1588fe960a1SDavid E. O'Brien 1598fe960a1SDavid E. O'Brien offsetof(Elf64_Shdr, sh_name), offsetof(Elf64_Shdr, sh_type), 1608fe960a1SDavid E. O'Brien offsetof(Elf64_Shdr, sh_flags), offsetof(Elf64_Shdr, sh_addr), 1618fe960a1SDavid E. O'Brien offsetof(Elf64_Shdr, sh_offset), offsetof(Elf64_Shdr, sh_size), 1628fe960a1SDavid E. O'Brien offsetof(Elf64_Shdr, sh_link), offsetof(Elf64_Shdr, sh_info), 1638fe960a1SDavid E. O'Brien offsetof(Elf64_Shdr, sh_addralign), offsetof(Elf64_Shdr, sh_entsize), 1648fe960a1SDavid E. O'Brien 1658fe960a1SDavid E. O'Brien offsetof(Elf64_Sym, st_name), offsetof(Elf64_Sym, st_value), 1668fe960a1SDavid E. O'Brien offsetof(Elf64_Sym, st_size), offsetof(Elf64_Sym, st_info), 1678fe960a1SDavid E. O'Brien offsetof(Elf64_Sym, st_shndx), 1688fe960a1SDavid E. O'Brien 1698fe960a1SDavid E. O'Brien offsetof(Elf64_Rel, r_offset), offsetof(Elf64_Rel, r_info), 1708fe960a1SDavid E. O'Brien 1718fe960a1SDavid E. O'Brien offsetof(Elf64_Rela, r_offset), offsetof(Elf64_Rela, r_info), 1728fe960a1SDavid E. O'Brien offsetof(Elf64_Rela, r_addend) 1738fe960a1SDavid E. O'Brien }; 1748fe960a1SDavid E. O'Brien 1753df3bb12SDavid E. O'Brien /* http://www.sco.com/developers/gabi/latest/ch5.dynamic.html#tag_encodings */ 17636210553SDavid E. O'Brien static const char * 1778edba452SAndreas Tobler d_tags(u_int64_t tag) 1788edba452SAndreas Tobler { 1798edba452SAndreas Tobler static char unknown_tag[48]; 1808edba452SAndreas Tobler 1813df3bb12SDavid E. O'Brien switch (tag) { 182ee8e9ecaSEd Maste case DT_NULL: return "DT_NULL"; 183ee8e9ecaSEd Maste case DT_NEEDED: return "DT_NEEDED"; 184ee8e9ecaSEd Maste case DT_PLTRELSZ: return "DT_PLTRELSZ"; 185ee8e9ecaSEd Maste case DT_PLTGOT: return "DT_PLTGOT"; 186ee8e9ecaSEd Maste case DT_HASH: return "DT_HASH"; 187ee8e9ecaSEd Maste case DT_STRTAB: return "DT_STRTAB"; 188ee8e9ecaSEd Maste case DT_SYMTAB: return "DT_SYMTAB"; 189ee8e9ecaSEd Maste case DT_RELA: return "DT_RELA"; 190ee8e9ecaSEd Maste case DT_RELASZ: return "DT_RELASZ"; 191ee8e9ecaSEd Maste case DT_RELAENT: return "DT_RELAENT"; 192ee8e9ecaSEd Maste case DT_STRSZ: return "DT_STRSZ"; 193ee8e9ecaSEd Maste case DT_SYMENT: return "DT_SYMENT"; 194ee8e9ecaSEd Maste case DT_INIT: return "DT_INIT"; 195ee8e9ecaSEd Maste case DT_FINI: return "DT_FINI"; 196ee8e9ecaSEd Maste case DT_SONAME: return "DT_SONAME"; 197ee8e9ecaSEd Maste case DT_RPATH: return "DT_RPATH"; 198ee8e9ecaSEd Maste case DT_SYMBOLIC: return "DT_SYMBOLIC"; 199ee8e9ecaSEd Maste case DT_REL: return "DT_REL"; 200ee8e9ecaSEd Maste case DT_RELSZ: return "DT_RELSZ"; 201ee8e9ecaSEd Maste case DT_RELENT: return "DT_RELENT"; 202ee8e9ecaSEd Maste case DT_PLTREL: return "DT_PLTREL"; 203ee8e9ecaSEd Maste case DT_DEBUG: return "DT_DEBUG"; 204ee8e9ecaSEd Maste case DT_TEXTREL: return "DT_TEXTREL"; 205ee8e9ecaSEd Maste case DT_JMPREL: return "DT_JMPREL"; 206ee8e9ecaSEd Maste case DT_BIND_NOW: return "DT_BIND_NOW"; 207ee8e9ecaSEd Maste case DT_INIT_ARRAY: return "DT_INIT_ARRAY"; 208ee8e9ecaSEd Maste case DT_FINI_ARRAY: return "DT_FINI_ARRAY"; 209ee8e9ecaSEd Maste case DT_INIT_ARRAYSZ: return "DT_INIT_ARRAYSZ"; 210ee8e9ecaSEd Maste case DT_FINI_ARRAYSZ: return "DT_FINI_ARRAYSZ"; 211ee8e9ecaSEd Maste case DT_RUNPATH: return "DT_RUNPATH"; 212ee8e9ecaSEd Maste case DT_FLAGS: return "DT_FLAGS"; 213ee8e9ecaSEd Maste case DT_PREINIT_ARRAY: return "DT_PREINIT_ARRAY"; /* XXX DT_ENCODING */ 214ee8e9ecaSEd Maste case DT_PREINIT_ARRAYSZ:return "DT_PREINIT_ARRAYSZ"; 2153df3bb12SDavid E. O'Brien /* 0x6000000D - 0x6ffff000 operating system-specific semantics */ 2163df3bb12SDavid E. O'Brien case 0x6ffffdf5: return "DT_GNU_PRELINKED"; 2173df3bb12SDavid E. O'Brien case 0x6ffffdf6: return "DT_GNU_CONFLICTSZ"; 2183df3bb12SDavid E. O'Brien case 0x6ffffdf7: return "DT_GNU_LIBLISTSZ"; 2193df3bb12SDavid E. O'Brien case 0x6ffffdf8: return "DT_SUNW_CHECKSUM"; 220ee8e9ecaSEd Maste case DT_PLTPADSZ: return "DT_PLTPADSZ"; 221ee8e9ecaSEd Maste case DT_MOVEENT: return "DT_MOVEENT"; 222ee8e9ecaSEd Maste case DT_MOVESZ: return "DT_MOVESZ"; 2234d52a0fdSEd Maste case DT_FEATURE: return "DT_FEATURE"; 224ee8e9ecaSEd Maste case DT_POSFLAG_1: return "DT_POSFLAG_1"; 225ee8e9ecaSEd Maste case DT_SYMINSZ: return "DT_SYMINSZ"; 226ee8e9ecaSEd Maste case DT_SYMINENT : return "DT_SYMINENT (DT_VALRNGHI)"; 227ee8e9ecaSEd Maste case DT_ADDRRNGLO: return "DT_ADDRRNGLO"; 228ee8e9ecaSEd Maste case DT_GNU_HASH: return "DT_GNU_HASH"; 2293df3bb12SDavid E. O'Brien case 0x6ffffef8: return "DT_GNU_CONFLICT"; 2303df3bb12SDavid E. O'Brien case 0x6ffffef9: return "DT_GNU_LIBLIST"; 231494053f4SEd Maste case DT_CONFIG: return "DT_CONFIG"; 232494053f4SEd Maste case DT_DEPAUDIT: return "DT_DEPAUDIT"; 233494053f4SEd Maste case DT_AUDIT: return "DT_AUDIT"; 234494053f4SEd Maste case DT_PLTPAD: return "DT_PLTPAD"; 235494053f4SEd Maste case DT_MOVETAB: return "DT_MOVETAB"; 236ee8e9ecaSEd Maste case DT_SYMINFO : return "DT_SYMINFO (DT_ADDRRNGHI)"; 237ee8e9ecaSEd Maste case DT_RELACOUNT: return "DT_RELACOUNT"; 238ee8e9ecaSEd Maste case DT_RELCOUNT: return "DT_RELCOUNT"; 239ee8e9ecaSEd Maste case DT_FLAGS_1: return "DT_FLAGS_1"; 240ee8e9ecaSEd Maste case DT_VERDEF: return "DT_VERDEF"; 241ee8e9ecaSEd Maste case DT_VERDEFNUM: return "DT_VERDEFNUM"; 242ee8e9ecaSEd Maste case DT_VERNEED: return "DT_VERNEED"; 243ee8e9ecaSEd Maste case DT_VERNEEDNUM: return "DT_VERNEEDNUM"; 2443df3bb12SDavid E. O'Brien case 0x6ffffff0: return "DT_GNU_VERSYM"; 2453df3bb12SDavid E. O'Brien /* 0x70000000 - 0x7fffffff processor-specific semantics */ 24605157fa0SMarcel Moolenaar case 0x70000000: return "DT_IA_64_PLT_RESERVE"; 247cf8fb39fSEd Maste case DT_AUXILIARY: return "DT_AUXILIARY"; 248cf8fb39fSEd Maste case DT_USED: return "DT_USED"; 249cf8fb39fSEd Maste case DT_FILTER: return "DT_FILTER"; 2503df3bb12SDavid E. O'Brien } 2518edba452SAndreas Tobler snprintf(unknown_tag, sizeof(unknown_tag), 2528edba452SAndreas Tobler "ERROR: TAG NOT DEFINED -- tag 0x%jx", (uintmax_t)tag); 2538edba452SAndreas Tobler return (unknown_tag); 25481f8caf7SJens Schweikhardt } 2558fe960a1SDavid E. O'Brien 25636210553SDavid E. O'Brien static const char * 2575f85bb36SMarcel Moolenaar e_machines(u_int mach) 2585f85bb36SMarcel Moolenaar { 25957668ad8SMarcel Moolenaar static char machdesc[64]; 26057668ad8SMarcel Moolenaar 2615f85bb36SMarcel Moolenaar switch (mach) { 2625f85bb36SMarcel Moolenaar case EM_NONE: return "EM_NONE"; 2635f85bb36SMarcel Moolenaar case EM_M32: return "EM_M32"; 2645f85bb36SMarcel Moolenaar case EM_SPARC: return "EM_SPARC"; 2655f85bb36SMarcel Moolenaar case EM_386: return "EM_386"; 2665f85bb36SMarcel Moolenaar case EM_68K: return "EM_68K"; 2675f85bb36SMarcel Moolenaar case EM_88K: return "EM_88K"; 2689afb8b24SEd Maste case EM_IAMCU: return "EM_IAMCU"; 2695f85bb36SMarcel Moolenaar case EM_860: return "EM_860"; 2705f85bb36SMarcel Moolenaar case EM_MIPS: return "EM_MIPS"; 27157668ad8SMarcel Moolenaar case EM_PPC: return "EM_PPC"; 272fee11555SJustin Hibbits case EM_PPC64: return "EM_PPC64"; 27357668ad8SMarcel Moolenaar case EM_ARM: return "EM_ARM"; 27457668ad8SMarcel Moolenaar case EM_ALPHA: return "EM_ALPHA (legacy)"; 27557668ad8SMarcel Moolenaar case EM_SPARCV9:return "EM_SPARCV9"; 2765f85bb36SMarcel Moolenaar case EM_IA_64: return "EM_IA_64"; 27757668ad8SMarcel Moolenaar case EM_X86_64: return "EM_X86_64"; 2782d538f8cSEd Maste case EM_AARCH64:return "EM_AARCH64"; 279119b7592SEd Maste case EM_RISCV: return "EM_RISCV"; 2805f85bb36SMarcel Moolenaar } 28157668ad8SMarcel Moolenaar snprintf(machdesc, sizeof(machdesc), 28257668ad8SMarcel Moolenaar "(unknown machine) -- type 0x%x", mach); 28357668ad8SMarcel Moolenaar return (machdesc); 28481f8caf7SJens Schweikhardt } 2858fe960a1SDavid E. O'Brien 286bf70beceSEd Schouten static const char *e_types[] = { 2878fe960a1SDavid E. O'Brien "ET_NONE", "ET_REL", "ET_EXEC", "ET_DYN", "ET_CORE" 2888fe960a1SDavid E. O'Brien }; 2898fe960a1SDavid E. O'Brien 290bf70beceSEd Schouten static const char *ei_versions[] = { 2918fe960a1SDavid E. O'Brien "EV_NONE", "EV_CURRENT" 2928fe960a1SDavid E. O'Brien }; 2938fe960a1SDavid E. O'Brien 294bf70beceSEd Schouten static const char *ei_classes[] = { 2958fe960a1SDavid E. O'Brien "ELFCLASSNONE", "ELFCLASS32", "ELFCLASS64" 2968fe960a1SDavid E. O'Brien }; 2978fe960a1SDavid E. O'Brien 298bf70beceSEd Schouten static const char *ei_data[] = { 2998fe960a1SDavid E. O'Brien "ELFDATANONE", "ELFDATA2LSB", "ELFDATA2MSB" 3008fe960a1SDavid E. O'Brien }; 3018fe960a1SDavid E. O'Brien 302bf70beceSEd Schouten static const char *ei_abis[256] = { 303453b09caSEd Maste "ELFOSABI_NONE", "ELFOSABI_HPUX", "ELFOSABI_NETBSD", "ELFOSABI_LINUX", 304ead61635SMarcel Moolenaar "ELFOSABI_HURD", "ELFOSABI_86OPEN", "ELFOSABI_SOLARIS", "ELFOSABI_AIX", 305ead61635SMarcel Moolenaar "ELFOSABI_IRIX", "ELFOSABI_FREEBSD", "ELFOSABI_TRU64", 306ead61635SMarcel Moolenaar "ELFOSABI_MODESTO", "ELFOSABI_OPENBSD", 307ead61635SMarcel Moolenaar [255] = "ELFOSABI_STANDALONE" 3088fe960a1SDavid E. O'Brien }; 3098fe960a1SDavid E. O'Brien 310bf70beceSEd Schouten static const char *p_types[] = { 3118fe960a1SDavid E. O'Brien "PT_NULL", "PT_LOAD", "PT_DYNAMIC", "PT_INTERP", "PT_NOTE", 312d499047cSJake Burkholder "PT_SHLIB", "PT_PHDR", "PT_TLS" 3138fe960a1SDavid E. O'Brien }; 3148fe960a1SDavid E. O'Brien 315bf70beceSEd Schouten static const char *p_flags[] = { 3168fe960a1SDavid E. O'Brien "", "PF_X", "PF_W", "PF_X|PF_W", "PF_R", "PF_X|PF_R", "PF_W|PF_R", 3178fe960a1SDavid E. O'Brien "PF_X|PF_W|PF_R" 3188fe960a1SDavid E. O'Brien }; 3198fe960a1SDavid E. O'Brien 320*97d368d6SEd Maste #define NT_ELEM(x) [x] = #x, 3212bc7b024SEd Maste static const char *nt_types[] = { 322*97d368d6SEd Maste "", 323*97d368d6SEd Maste NT_ELEM(NT_FREEBSD_ABI_TAG) 324*97d368d6SEd Maste NT_ELEM(NT_FREEBSD_NOINIT_TAG) 325*97d368d6SEd Maste NT_ELEM(NT_FREEBSD_ARCH_TAG) 326*97d368d6SEd Maste NT_ELEM(NT_FREEBSD_FEATURE_CTL) 3272bc7b024SEd Maste }; 3282bc7b024SEd Maste 3299bc01124SDavid E. O'Brien /* http://www.sco.com/developers/gabi/latest/ch4.sheader.html#sh_type */ 33036210553SDavid E. O'Brien static const char * 331b3854059SEd Maste sh_types(uint64_t machine, uint64_t sht) { 332b3854059SEd Maste static char unknown_buf[64]; 333b3854059SEd Maste 334b3854059SEd Maste if (sht < 0x60000000) { 3359bc01124SDavid E. O'Brien switch (sht) { 336ee8e9ecaSEd Maste case SHT_NULL: return "SHT_NULL"; 337ee8e9ecaSEd Maste case SHT_PROGBITS: return "SHT_PROGBITS"; 338ee8e9ecaSEd Maste case SHT_SYMTAB: return "SHT_SYMTAB"; 339ee8e9ecaSEd Maste case SHT_STRTAB: return "SHT_STRTAB"; 340ee8e9ecaSEd Maste case SHT_RELA: return "SHT_RELA"; 341ee8e9ecaSEd Maste case SHT_HASH: return "SHT_HASH"; 342ee8e9ecaSEd Maste case SHT_DYNAMIC: return "SHT_DYNAMIC"; 343ee8e9ecaSEd Maste case SHT_NOTE: return "SHT_NOTE"; 344ee8e9ecaSEd Maste case SHT_NOBITS: return "SHT_NOBITS"; 345ee8e9ecaSEd Maste case SHT_REL: return "SHT_REL"; 346ee8e9ecaSEd Maste case SHT_SHLIB: return "SHT_SHLIB"; 347ee8e9ecaSEd Maste case SHT_DYNSYM: return "SHT_DYNSYM"; 348ee8e9ecaSEd Maste case SHT_INIT_ARRAY: return "SHT_INIT_ARRAY"; 349ee8e9ecaSEd Maste case SHT_FINI_ARRAY: return "SHT_FINI_ARRAY"; 350ee8e9ecaSEd Maste case SHT_PREINIT_ARRAY: return "SHT_PREINIT_ARRAY"; 351ee8e9ecaSEd Maste case SHT_GROUP: return "SHT_GROUP"; 352ee8e9ecaSEd Maste case SHT_SYMTAB_SHNDX: return "SHT_SYMTAB_SHNDX"; 353b3854059SEd Maste } 354b3854059SEd Maste snprintf(unknown_buf, sizeof(unknown_buf), 355b3854059SEd Maste "ERROR: SHT %ju NOT DEFINED", (uintmax_t)sht); 356b3854059SEd Maste return (unknown_buf); 357b3854059SEd Maste } else if (sht < 0x70000000) { 3583df3bb12SDavid E. O'Brien /* 0x60000000-0x6fffffff operating system-specific semantics */ 359b3854059SEd Maste switch (sht) { 3609bc01124SDavid E. O'Brien case 0x6ffffff0: return "XXX:VERSYM"; 361ee8e9ecaSEd Maste case SHT_SUNW_dof: return "SHT_SUNW_dof"; 362ee8e9ecaSEd Maste case SHT_GNU_HASH: return "SHT_GNU_HASH"; 3639bc01124SDavid E. O'Brien case 0x6ffffff7: return "SHT_GNU_LIBLIST"; 3649bc01124SDavid E. O'Brien case 0x6ffffffc: return "XXX:VERDEF"; 365ee8e9ecaSEd Maste case SHT_SUNW_verdef: return "SHT_SUNW(GNU)_verdef"; 366ee8e9ecaSEd Maste case SHT_SUNW_verneed: return "SHT_SUNW(GNU)_verneed"; 367ee8e9ecaSEd Maste case SHT_SUNW_versym: return "SHT_SUNW(GNU)_versym"; 368b3854059SEd Maste } 369b3854059SEd Maste snprintf(unknown_buf, sizeof(unknown_buf), 370b3854059SEd Maste "ERROR: OS-SPECIFIC SHT 0x%jx NOT DEFINED", 371b3854059SEd Maste (uintmax_t)sht); 372b3854059SEd Maste return (unknown_buf); 373b3854059SEd Maste } else if (sht < 0x80000000) { 3743df3bb12SDavid E. O'Brien /* 0x70000000-0x7fffffff processor-specific semantics */ 375b3854059SEd Maste switch (machine) { 3768edba452SAndreas Tobler case EM_ARM: 3778edba452SAndreas Tobler switch (sht) { 378ee8e9ecaSEd Maste case SHT_ARM_EXIDX: return "SHT_ARM_EXIDX"; 379ee8e9ecaSEd Maste case SHT_ARM_PREEMPTMAP:return "SHT_ARM_PREEMPTMAP"; 380ee8e9ecaSEd Maste case SHT_ARM_ATTRIBUTES:return "SHT_ARM_ATTRIBUTES"; 381ee8e9ecaSEd Maste case SHT_ARM_DEBUGOVERLAY: 382ee8e9ecaSEd Maste return "SHT_ARM_DEBUGOVERLAY"; 383ee8e9ecaSEd Maste case SHT_ARM_OVERLAYSECTION: 384ee8e9ecaSEd Maste return "SHT_ARM_OVERLAYSECTION"; 385b3854059SEd Maste } 386b3854059SEd Maste break; 387b3854059SEd Maste case EM_IA_64: 388b3854059SEd Maste switch (sht) { 3895f85bb36SMarcel Moolenaar case 0x70000000: return "SHT_IA_64_EXT"; 3905f85bb36SMarcel Moolenaar case 0x70000001: return "SHT_IA_64_UNWIND"; 391b3854059SEd Maste } 392b3854059SEd Maste break; 393ee8e9ecaSEd Maste case EM_MIPS: 394ee8e9ecaSEd Maste switch (sht) { 395050747f2SEd Maste case SHT_MIPS_REGINFO: return "SHT_MIPS_REGINFO"; 396ee8e9ecaSEd Maste case SHT_MIPS_OPTIONS: return "SHT_MIPS_OPTIONS"; 3974db90148SEd Maste case SHT_MIPS_ABIFLAGS: return "SHT_MIPS_ABIFLAGS"; 398ee8e9ecaSEd Maste } 399ee8e9ecaSEd Maste break; 400b3854059SEd Maste } 401b3854059SEd Maste switch (sht) { 4029bc01124SDavid E. O'Brien case 0x7ffffffd: return "XXX:AUXILIARY"; 4039bc01124SDavid E. O'Brien case 0x7fffffff: return "XXX:FILTER"; 404b3854059SEd Maste } 405b3854059SEd Maste snprintf(unknown_buf, sizeof(unknown_buf), 406b3854059SEd Maste "ERROR: PROCESSOR-SPECIFIC SHT 0x%jx NOT DEFINED", 407b3854059SEd Maste (uintmax_t)sht); 408b3854059SEd Maste return (unknown_buf); 409b3854059SEd Maste } else { 4103df3bb12SDavid E. O'Brien /* 0x80000000-0xffffffff application programs */ 411b3854059SEd Maste snprintf(unknown_buf, sizeof(unknown_buf), 412b3854059SEd Maste "ERROR: SHT 0x%jx NOT DEFINED", 413b3854059SEd Maste (uintmax_t)sht); 414b3854059SEd Maste return (unknown_buf); 4159bc01124SDavid E. O'Brien } 416dcae6166SJens Schweikhardt } 4178fe960a1SDavid E. O'Brien 418bf70beceSEd Schouten static const char *sh_flags[] = { 4198fe960a1SDavid E. O'Brien "", "SHF_WRITE", "SHF_ALLOC", "SHF_WRITE|SHF_ALLOC", "SHF_EXECINSTR", 4208fe960a1SDavid E. O'Brien "SHF_WRITE|SHF_EXECINSTR", "SHF_ALLOC|SHF_EXECINSTR", 4218fe960a1SDavid E. O'Brien "SHF_WRITE|SHF_ALLOC|SHF_EXECINSTR" 4228fe960a1SDavid E. O'Brien }; 4238fe960a1SDavid E. O'Brien 4243a4b59a2SEd Maste static const char * 4253a4b59a2SEd Maste st_type(unsigned int mach, unsigned int type) 4263a4b59a2SEd Maste { 4273a4b59a2SEd Maste static char s_type[32]; 4283a4b59a2SEd Maste 4293a4b59a2SEd Maste switch (type) { 4303a4b59a2SEd Maste case STT_NOTYPE: return "STT_NOTYPE"; 4313a4b59a2SEd Maste case STT_OBJECT: return "STT_OBJECT"; 4323a4b59a2SEd Maste case STT_FUNC: return "STT_FUNC"; 4333a4b59a2SEd Maste case STT_SECTION: return "STT_SECTION"; 4343a4b59a2SEd Maste case STT_FILE: return "STT_FILE"; 4353a4b59a2SEd Maste case STT_COMMON: return "STT_COMMON"; 4363a4b59a2SEd Maste case STT_TLS: return "STT_TLS"; 4373a4b59a2SEd Maste case 13: 4383a4b59a2SEd Maste if (mach == EM_SPARCV9) 4393a4b59a2SEd Maste return "STT_SPARC_REGISTER"; 4403a4b59a2SEd Maste break; 4413a4b59a2SEd Maste } 4423a4b59a2SEd Maste snprintf(s_type, sizeof(s_type), "<unknown: %#x>", type); 4433a4b59a2SEd Maste return (s_type); 4443a4b59a2SEd Maste } 4458fe960a1SDavid E. O'Brien 446bf70beceSEd Schouten static const char *st_bindings[] = { 4478fe960a1SDavid E. O'Brien "STB_LOCAL", "STB_GLOBAL", "STB_WEAK" 4488fe960a1SDavid E. O'Brien }; 4498fe960a1SDavid E. O'Brien 450bf70beceSEd Schouten static char *dynstr; 451bf70beceSEd Schouten static char *shstrtab; 452bf70beceSEd Schouten static char *strtab; 453bf70beceSEd Schouten static FILE *out; 4548fe960a1SDavid E. O'Brien 455bf70beceSEd Schouten static u_int64_t elf_get_byte(Elf32_Ehdr *e, void *base, elf_member_t member); 456bf70beceSEd Schouten static u_int64_t elf_get_quarter(Elf32_Ehdr *e, void *base, 457bf70beceSEd Schouten elf_member_t member); 458bf70beceSEd Schouten #if 0 459bf70beceSEd Schouten static u_int64_t elf_get_half(Elf32_Ehdr *e, void *base, elf_member_t member); 460bf70beceSEd Schouten #endif 461bf70beceSEd Schouten static u_int64_t elf_get_word(Elf32_Ehdr *e, void *base, elf_member_t member); 462bf70beceSEd Schouten static u_int64_t elf_get_quad(Elf32_Ehdr *e, void *base, elf_member_t member); 4638fe960a1SDavid E. O'Brien 46497a9b0b1SEd Maste static void elf_print_ehdr(Elf32_Ehdr *e, void *sh); 465bf70beceSEd Schouten static void elf_print_phdr(Elf32_Ehdr *e, void *p); 466bf70beceSEd Schouten static void elf_print_shdr(Elf32_Ehdr *e, void *sh); 467bf70beceSEd Schouten static void elf_print_symtab(Elf32_Ehdr *e, void *sh, char *str); 468bf70beceSEd Schouten static void elf_print_dynamic(Elf32_Ehdr *e, void *sh); 469bf70beceSEd Schouten static void elf_print_rel(Elf32_Ehdr *e, void *r); 470bf70beceSEd Schouten static void elf_print_rela(Elf32_Ehdr *e, void *ra); 471bf70beceSEd Schouten static void elf_print_interp(Elf32_Ehdr *e, void *p); 472bf70beceSEd Schouten static void elf_print_got(Elf32_Ehdr *e, void *sh); 473bf70beceSEd Schouten static void elf_print_hash(Elf32_Ehdr *e, void *sh); 474bf70beceSEd Schouten static void elf_print_note(Elf32_Ehdr *e, void *sh); 4758fe960a1SDavid E. O'Brien 476bf70beceSEd Schouten static void usage(void); 4778fe960a1SDavid E. O'Brien 47897a9b0b1SEd Maste /* 47997a9b0b1SEd Maste * Helpers for ELF files with shnum or shstrndx values that don't fit in the 48097a9b0b1SEd Maste * ELF header. If the values are too large then an escape value is used to 48197a9b0b1SEd Maste * indicate that the actual value is found in one of section 0's fields. 48297a9b0b1SEd Maste */ 48397a9b0b1SEd Maste static uint64_t 48497a9b0b1SEd Maste elf_get_shnum(Elf32_Ehdr *e, void *sh) 48597a9b0b1SEd Maste { 48697a9b0b1SEd Maste uint64_t shnum; 48797a9b0b1SEd Maste 48897a9b0b1SEd Maste shnum = elf_get_quarter(e, e, E_SHNUM); 48997a9b0b1SEd Maste if (shnum == 0) 49097a9b0b1SEd Maste shnum = elf_get_word(e, (char *)sh, SH_SIZE); 49197a9b0b1SEd Maste return shnum; 49297a9b0b1SEd Maste } 49397a9b0b1SEd Maste 49497a9b0b1SEd Maste static uint64_t 49597a9b0b1SEd Maste elf_get_shstrndx(Elf32_Ehdr *e, void *sh) 49697a9b0b1SEd Maste { 49797a9b0b1SEd Maste uint64_t shstrndx; 49897a9b0b1SEd Maste 49997a9b0b1SEd Maste shstrndx = elf_get_quarter(e, e, E_SHSTRNDX); 50097a9b0b1SEd Maste if (shstrndx == SHN_XINDEX) 50197a9b0b1SEd Maste shstrndx = elf_get_word(e, (char *)sh, SH_LINK); 50297a9b0b1SEd Maste return shstrndx; 50397a9b0b1SEd Maste } 50497a9b0b1SEd Maste 5058fe960a1SDavid E. O'Brien int 5068fe960a1SDavid E. O'Brien main(int ac, char **av) 5078fe960a1SDavid E. O'Brien { 5089efc7e72SChristian Brueffer cap_rights_t rights; 5098fe960a1SDavid E. O'Brien u_int64_t phoff; 5108fe960a1SDavid E. O'Brien u_int64_t shoff; 5118fe960a1SDavid E. O'Brien u_int64_t phentsize; 5128fe960a1SDavid E. O'Brien u_int64_t phnum; 5138fe960a1SDavid E. O'Brien u_int64_t shentsize; 5148fe960a1SDavid E. O'Brien u_int64_t shnum; 5158fe960a1SDavid E. O'Brien u_int64_t shstrndx; 5168fe960a1SDavid E. O'Brien u_int64_t offset; 5178fe960a1SDavid E. O'Brien u_int64_t name; 5188fe960a1SDavid E. O'Brien u_int64_t type; 5198fe960a1SDavid E. O'Brien struct stat sb; 5208fe960a1SDavid E. O'Brien u_int flags; 5210529c56bSDavid E. O'Brien Elf32_Ehdr *e; 5228fe960a1SDavid E. O'Brien void *p; 5238fe960a1SDavid E. O'Brien void *sh; 5248fe960a1SDavid E. O'Brien void *v; 5258fe960a1SDavid E. O'Brien int fd; 5268fe960a1SDavid E. O'Brien int ch; 5278fe960a1SDavid E. O'Brien int i; 5288fe960a1SDavid E. O'Brien 5298fe960a1SDavid E. O'Brien out = stdout; 5308fe960a1SDavid E. O'Brien flags = 0; 531c35530f4SBrooks Davis while ((ch = getopt(ac, av, "acdEeiGhnprsw:")) != -1) 5328fe960a1SDavid E. O'Brien switch (ch) { 5338fe960a1SDavid E. O'Brien case 'a': 5348fe960a1SDavid E. O'Brien flags = ED_ALL; 5358fe960a1SDavid E. O'Brien break; 5368fe960a1SDavid E. O'Brien case 'c': 5378fe960a1SDavid E. O'Brien flags |= ED_SHDR; 5388fe960a1SDavid E. O'Brien break; 5398fe960a1SDavid E. O'Brien case 'd': 5408fe960a1SDavid E. O'Brien flags |= ED_DYN; 5418fe960a1SDavid E. O'Brien break; 542c35530f4SBrooks Davis case 'E': 543c35530f4SBrooks Davis flags = ED_IS_ELF; 544c35530f4SBrooks Davis break; 5458fe960a1SDavid E. O'Brien case 'e': 5468fe960a1SDavid E. O'Brien flags |= ED_EHDR; 5478fe960a1SDavid E. O'Brien break; 5488fe960a1SDavid E. O'Brien case 'i': 5498fe960a1SDavid E. O'Brien flags |= ED_INTERP; 5508fe960a1SDavid E. O'Brien break; 5518fe960a1SDavid E. O'Brien case 'G': 5528fe960a1SDavid E. O'Brien flags |= ED_GOT; 5538fe960a1SDavid E. O'Brien break; 5548fe960a1SDavid E. O'Brien case 'h': 5558fe960a1SDavid E. O'Brien flags |= ED_HASH; 5568fe960a1SDavid E. O'Brien break; 5578fe960a1SDavid E. O'Brien case 'n': 5588fe960a1SDavid E. O'Brien flags |= ED_NOTE; 5598fe960a1SDavid E. O'Brien break; 5608fe960a1SDavid E. O'Brien case 'p': 5618fe960a1SDavid E. O'Brien flags |= ED_PHDR; 5628fe960a1SDavid E. O'Brien break; 5638fe960a1SDavid E. O'Brien case 'r': 5648fe960a1SDavid E. O'Brien flags |= ED_REL; 5658fe960a1SDavid E. O'Brien break; 5668fe960a1SDavid E. O'Brien case 's': 5678fe960a1SDavid E. O'Brien flags |= ED_SYMTAB; 5688fe960a1SDavid E. O'Brien break; 5698fe960a1SDavid E. O'Brien case 'w': 5708fe960a1SDavid E. O'Brien if ((out = fopen(optarg, "w")) == NULL) 5718fe960a1SDavid E. O'Brien err(1, "%s", optarg); 5729efc7e72SChristian Brueffer cap_rights_init(&rights, CAP_FSTAT, CAP_WRITE); 573377421dfSMariusz Zaborski if (caph_rights_limit(fileno(out), &rights) < 0) 5749efc7e72SChristian Brueffer err(1, "unable to limit rights for %s", optarg); 5758fe960a1SDavid E. O'Brien break; 5768fe960a1SDavid E. O'Brien case '?': 5778fe960a1SDavid E. O'Brien default: 5788fe960a1SDavid E. O'Brien usage(); 5798fe960a1SDavid E. O'Brien } 5808fe960a1SDavid E. O'Brien ac -= optind; 5818fe960a1SDavid E. O'Brien av += optind; 582c35530f4SBrooks Davis if (ac == 0 || flags == 0 || ((flags & ED_IS_ELF) && 583c35530f4SBrooks Davis (ac != 1 || (flags & ~ED_IS_ELF) || out != stdout))) 5848fe960a1SDavid E. O'Brien usage(); 5858fe960a1SDavid E. O'Brien if ((fd = open(*av, O_RDONLY)) < 0 || 5868fe960a1SDavid E. O'Brien fstat(fd, &sb) < 0) 58773e57b80SRuslan Ermilov err(1, "%s", *av); 5889efc7e72SChristian Brueffer cap_rights_init(&rights, CAP_MMAP_R); 589377421dfSMariusz Zaborski if (caph_rights_limit(fd, &rights) < 0) 5909efc7e72SChristian Brueffer err(1, "unable to limit rights for %s", *av); 591a4e3fc54SMariusz Zaborski cap_rights_init(&rights); 592377421dfSMariusz Zaborski if (caph_rights_limit(STDIN_FILENO, &rights) < 0 || 593a4e3fc54SMariusz Zaborski caph_limit_stdout() < 0 || caph_limit_stderr() < 0) { 594a4e3fc54SMariusz Zaborski err(1, "unable to limit rights for stdio"); 595a4e3fc54SMariusz Zaborski } 5967672a014SMariusz Zaborski if (caph_enter() < 0) 5979efc7e72SChristian Brueffer err(1, "unable to enter capability mode"); 5988fe960a1SDavid E. O'Brien e = mmap(NULL, sb.st_size, PROT_READ, MAP_SHARED, fd, 0); 5998fe960a1SDavid E. O'Brien if (e == MAP_FAILED) 6008fe960a1SDavid E. O'Brien err(1, NULL); 601c35530f4SBrooks Davis if (!IS_ELF(*(Elf32_Ehdr *)e)) { 602c35530f4SBrooks Davis if (flags & ED_IS_ELF) 603c35530f4SBrooks Davis exit(1); 6048fe960a1SDavid E. O'Brien errx(1, "not an elf file"); 605c35530f4SBrooks Davis } else if (flags & ED_IS_ELF) 606c35530f4SBrooks Davis exit (0); 6078fe960a1SDavid E. O'Brien phoff = elf_get_off(e, e, E_PHOFF); 6088fe960a1SDavid E. O'Brien shoff = elf_get_off(e, e, E_SHOFF); 6098fe960a1SDavid E. O'Brien phentsize = elf_get_quarter(e, e, E_PHENTSIZE); 6108fe960a1SDavid E. O'Brien phnum = elf_get_quarter(e, e, E_PHNUM); 6118fe960a1SDavid E. O'Brien shentsize = elf_get_quarter(e, e, E_SHENTSIZE); 61236210553SDavid E. O'Brien p = (char *)e + phoff; 613a589e596SMarcel Moolenaar if (shoff > 0) { 61436210553SDavid E. O'Brien sh = (char *)e + shoff; 61597a9b0b1SEd Maste shnum = elf_get_shnum(e, sh); 61697a9b0b1SEd Maste shstrndx = elf_get_shstrndx(e, sh); 617a589e596SMarcel Moolenaar offset = elf_get_off(e, (char *)sh + shstrndx * shentsize, 618a589e596SMarcel Moolenaar SH_OFFSET); 6190529c56bSDavid E. O'Brien shstrtab = (char *)e + offset; 620a589e596SMarcel Moolenaar } else { 621a589e596SMarcel Moolenaar sh = NULL; 622a589e596SMarcel Moolenaar shnum = 0; 623a589e596SMarcel Moolenaar shstrndx = 0; 624a589e596SMarcel Moolenaar shstrtab = NULL; 625a589e596SMarcel Moolenaar } 62636210553SDavid E. O'Brien for (i = 0; (u_int64_t)i < shnum; i++) { 62736210553SDavid E. O'Brien name = elf_get_word(e, (char *)sh + i * shentsize, SH_NAME); 62836210553SDavid E. O'Brien offset = elf_get_off(e, (char *)sh + i * shentsize, SH_OFFSET); 6298fe960a1SDavid E. O'Brien if (strcmp(shstrtab + name, ".strtab") == 0) 6300529c56bSDavid E. O'Brien strtab = (char *)e + offset; 6318fe960a1SDavid E. O'Brien if (strcmp(shstrtab + name, ".dynstr") == 0) 6320529c56bSDavid E. O'Brien dynstr = (char *)e + offset; 6338fe960a1SDavid E. O'Brien } 6348fe960a1SDavid E. O'Brien if (flags & ED_EHDR) 63597a9b0b1SEd Maste elf_print_ehdr(e, sh); 6368fe960a1SDavid E. O'Brien if (flags & ED_PHDR) 6378fe960a1SDavid E. O'Brien elf_print_phdr(e, p); 6388fe960a1SDavid E. O'Brien if (flags & ED_SHDR) 6398fe960a1SDavid E. O'Brien elf_print_shdr(e, sh); 64036210553SDavid E. O'Brien for (i = 0; (u_int64_t)i < phnum; i++) { 64136210553SDavid E. O'Brien v = (char *)p + i * phentsize; 6428fe960a1SDavid E. O'Brien type = elf_get_word(e, v, P_TYPE); 6438fe960a1SDavid E. O'Brien switch (type) { 6448fe960a1SDavid E. O'Brien case PT_INTERP: 6458fe960a1SDavid E. O'Brien if (flags & ED_INTERP) 6468fe960a1SDavid E. O'Brien elf_print_interp(e, v); 6478fe960a1SDavid E. O'Brien break; 6488fe960a1SDavid E. O'Brien case PT_NULL: 6498fe960a1SDavid E. O'Brien case PT_LOAD: 6508fe960a1SDavid E. O'Brien case PT_DYNAMIC: 6518fe960a1SDavid E. O'Brien case PT_NOTE: 6528fe960a1SDavid E. O'Brien case PT_SHLIB: 6538fe960a1SDavid E. O'Brien case PT_PHDR: 6548fe960a1SDavid E. O'Brien break; 6558fe960a1SDavid E. O'Brien } 6568fe960a1SDavid E. O'Brien } 65736210553SDavid E. O'Brien for (i = 0; (u_int64_t)i < shnum; i++) { 65836210553SDavid E. O'Brien v = (char *)sh + i * shentsize; 6598fe960a1SDavid E. O'Brien type = elf_get_word(e, v, SH_TYPE); 6608fe960a1SDavid E. O'Brien switch (type) { 6618fe960a1SDavid E. O'Brien case SHT_SYMTAB: 6628fe960a1SDavid E. O'Brien if (flags & ED_SYMTAB) 6638fe960a1SDavid E. O'Brien elf_print_symtab(e, v, strtab); 6648fe960a1SDavid E. O'Brien break; 6658fe960a1SDavid E. O'Brien case SHT_DYNAMIC: 6668fe960a1SDavid E. O'Brien if (flags & ED_DYN) 6678fe960a1SDavid E. O'Brien elf_print_dynamic(e, v); 6688fe960a1SDavid E. O'Brien break; 6698fe960a1SDavid E. O'Brien case SHT_RELA: 6708fe960a1SDavid E. O'Brien if (flags & ED_REL) 6718fe960a1SDavid E. O'Brien elf_print_rela(e, v); 6728fe960a1SDavid E. O'Brien break; 6738fe960a1SDavid E. O'Brien case SHT_REL: 6748fe960a1SDavid E. O'Brien if (flags & ED_REL) 6758fe960a1SDavid E. O'Brien elf_print_rel(e, v); 6768fe960a1SDavid E. O'Brien break; 6778fe960a1SDavid E. O'Brien case SHT_NOTE: 6788fe960a1SDavid E. O'Brien name = elf_get_word(e, v, SH_NAME); 6798fe960a1SDavid E. O'Brien if (flags & ED_NOTE && 680f37e7f52SEd Maste strcmp(shstrtab + name, ".note.tag") == 0) 6818fe960a1SDavid E. O'Brien elf_print_note(e, v); 6828fe960a1SDavid E. O'Brien break; 6838fe960a1SDavid E. O'Brien case SHT_DYNSYM: 6848fe960a1SDavid E. O'Brien if (flags & ED_SYMTAB) 6858fe960a1SDavid E. O'Brien elf_print_symtab(e, v, dynstr); 6868fe960a1SDavid E. O'Brien break; 6878fe960a1SDavid E. O'Brien case SHT_PROGBITS: 6888fe960a1SDavid E. O'Brien name = elf_get_word(e, v, SH_NAME); 6898fe960a1SDavid E. O'Brien if (flags & ED_GOT && 6908fe960a1SDavid E. O'Brien strcmp(shstrtab + name, ".got") == 0) 6918fe960a1SDavid E. O'Brien elf_print_got(e, v); 6928fe960a1SDavid E. O'Brien break; 6938fe960a1SDavid E. O'Brien case SHT_HASH: 6948fe960a1SDavid E. O'Brien if (flags & ED_HASH) 6958fe960a1SDavid E. O'Brien elf_print_hash(e, v); 6968fe960a1SDavid E. O'Brien break; 6978fe960a1SDavid E. O'Brien case SHT_NULL: 6988fe960a1SDavid E. O'Brien case SHT_STRTAB: 6998fe960a1SDavid E. O'Brien case SHT_NOBITS: 7008fe960a1SDavid E. O'Brien case SHT_SHLIB: 7018fe960a1SDavid E. O'Brien break; 7028fe960a1SDavid E. O'Brien } 7038fe960a1SDavid E. O'Brien } 7048fe960a1SDavid E. O'Brien 7058fe960a1SDavid E. O'Brien return 0; 7068fe960a1SDavid E. O'Brien } 7078fe960a1SDavid E. O'Brien 708bf70beceSEd Schouten static void 70997a9b0b1SEd Maste elf_print_ehdr(Elf32_Ehdr *e, void *sh) 7108fe960a1SDavid E. O'Brien { 7118fe960a1SDavid E. O'Brien u_int64_t class; 7128fe960a1SDavid E. O'Brien u_int64_t data; 7138fe960a1SDavid E. O'Brien u_int64_t osabi; 7148fe960a1SDavid E. O'Brien u_int64_t type; 7158fe960a1SDavid E. O'Brien u_int64_t machine; 7168fe960a1SDavid E. O'Brien u_int64_t version; 7178fe960a1SDavid E. O'Brien u_int64_t entry; 7188fe960a1SDavid E. O'Brien u_int64_t phoff; 7198fe960a1SDavid E. O'Brien u_int64_t shoff; 7208fe960a1SDavid E. O'Brien u_int64_t flags; 7218fe960a1SDavid E. O'Brien u_int64_t ehsize; 7228fe960a1SDavid E. O'Brien u_int64_t phentsize; 7238fe960a1SDavid E. O'Brien u_int64_t phnum; 7248fe960a1SDavid E. O'Brien u_int64_t shentsize; 7258fe960a1SDavid E. O'Brien u_int64_t shnum; 7268fe960a1SDavid E. O'Brien u_int64_t shstrndx; 7278fe960a1SDavid E. O'Brien 7288fe960a1SDavid E. O'Brien class = elf_get_byte(e, e, E_CLASS); 7298fe960a1SDavid E. O'Brien data = elf_get_byte(e, e, E_DATA); 7308fe960a1SDavid E. O'Brien osabi = elf_get_byte(e, e, E_OSABI); 7318fe960a1SDavid E. O'Brien type = elf_get_quarter(e, e, E_TYPE); 7328fe960a1SDavid E. O'Brien machine = elf_get_quarter(e, e, E_MACHINE); 7338fe960a1SDavid E. O'Brien version = elf_get_word(e, e, E_VERSION); 7348fe960a1SDavid E. O'Brien entry = elf_get_addr(e, e, E_ENTRY); 7358fe960a1SDavid E. O'Brien phoff = elf_get_off(e, e, E_PHOFF); 7368fe960a1SDavid E. O'Brien shoff = elf_get_off(e, e, E_SHOFF); 7378fe960a1SDavid E. O'Brien flags = elf_get_word(e, e, E_FLAGS); 7388fe960a1SDavid E. O'Brien ehsize = elf_get_quarter(e, e, E_EHSIZE); 7398fe960a1SDavid E. O'Brien phentsize = elf_get_quarter(e, e, E_PHENTSIZE); 7408fe960a1SDavid E. O'Brien phnum = elf_get_quarter(e, e, E_PHNUM); 7418fe960a1SDavid E. O'Brien shentsize = elf_get_quarter(e, e, E_SHENTSIZE); 7428fe960a1SDavid E. O'Brien fprintf(out, "\nelf header:\n"); 7438fe960a1SDavid E. O'Brien fprintf(out, "\n"); 7448fe960a1SDavid E. O'Brien fprintf(out, "\te_ident: %s %s %s\n", ei_classes[class], ei_data[data], 7458fe960a1SDavid E. O'Brien ei_abis[osabi]); 7468fe960a1SDavid E. O'Brien fprintf(out, "\te_type: %s\n", e_types[type]); 7475f85bb36SMarcel Moolenaar fprintf(out, "\te_machine: %s\n", e_machines(machine)); 7488fe960a1SDavid E. O'Brien fprintf(out, "\te_version: %s\n", ei_versions[version]); 74975ca4dcfSDavid E. O'Brien fprintf(out, "\te_entry: %#jx\n", (intmax_t)entry); 75075ca4dcfSDavid E. O'Brien fprintf(out, "\te_phoff: %jd\n", (intmax_t)phoff); 75175ca4dcfSDavid E. O'Brien fprintf(out, "\te_shoff: %jd\n", (intmax_t)shoff); 75275ca4dcfSDavid E. O'Brien fprintf(out, "\te_flags: %jd\n", (intmax_t)flags); 75375ca4dcfSDavid E. O'Brien fprintf(out, "\te_ehsize: %jd\n", (intmax_t)ehsize); 75475ca4dcfSDavid E. O'Brien fprintf(out, "\te_phentsize: %jd\n", (intmax_t)phentsize); 75575ca4dcfSDavid E. O'Brien fprintf(out, "\te_phnum: %jd\n", (intmax_t)phnum); 75675ca4dcfSDavid E. O'Brien fprintf(out, "\te_shentsize: %jd\n", (intmax_t)shentsize); 757a589e596SMarcel Moolenaar if (sh != NULL) { 758a589e596SMarcel Moolenaar shnum = elf_get_shnum(e, sh); 759a589e596SMarcel Moolenaar shstrndx = elf_get_shstrndx(e, sh); 76075ca4dcfSDavid E. O'Brien fprintf(out, "\te_shnum: %jd\n", (intmax_t)shnum); 76175ca4dcfSDavid E. O'Brien fprintf(out, "\te_shstrndx: %jd\n", (intmax_t)shstrndx); 7628fe960a1SDavid E. O'Brien } 763a589e596SMarcel Moolenaar } 7648fe960a1SDavid E. O'Brien 765bf70beceSEd Schouten static void 7660529c56bSDavid E. O'Brien elf_print_phdr(Elf32_Ehdr *e, void *p) 7678fe960a1SDavid E. O'Brien { 7688fe960a1SDavid E. O'Brien u_int64_t phentsize; 7698fe960a1SDavid E. O'Brien u_int64_t phnum; 7708fe960a1SDavid E. O'Brien u_int64_t type; 7718fe960a1SDavid E. O'Brien u_int64_t offset; 7728fe960a1SDavid E. O'Brien u_int64_t vaddr; 7738fe960a1SDavid E. O'Brien u_int64_t paddr; 7748fe960a1SDavid E. O'Brien u_int64_t filesz; 7758fe960a1SDavid E. O'Brien u_int64_t memsz; 7768fe960a1SDavid E. O'Brien u_int64_t flags; 7778fe960a1SDavid E. O'Brien u_int64_t align; 7788fe960a1SDavid E. O'Brien void *v; 7798fe960a1SDavid E. O'Brien int i; 7808fe960a1SDavid E. O'Brien 7818fe960a1SDavid E. O'Brien phentsize = elf_get_quarter(e, e, E_PHENTSIZE); 7828fe960a1SDavid E. O'Brien phnum = elf_get_quarter(e, e, E_PHNUM); 7838fe960a1SDavid E. O'Brien fprintf(out, "\nprogram header:\n"); 78436210553SDavid E. O'Brien for (i = 0; (u_int64_t)i < phnum; i++) { 78536210553SDavid E. O'Brien v = (char *)p + i * phentsize; 7868fe960a1SDavid E. O'Brien type = elf_get_word(e, v, P_TYPE); 7878fe960a1SDavid E. O'Brien offset = elf_get_off(e, v, P_OFFSET); 7888fe960a1SDavid E. O'Brien vaddr = elf_get_addr(e, v, P_VADDR); 7898fe960a1SDavid E. O'Brien paddr = elf_get_addr(e, v, P_PADDR); 7908fe960a1SDavid E. O'Brien filesz = elf_get_size(e, v, P_FILESZ); 7918fe960a1SDavid E. O'Brien memsz = elf_get_size(e, v, P_MEMSZ); 7928fe960a1SDavid E. O'Brien flags = elf_get_word(e, v, P_FLAGS); 7938fe960a1SDavid E. O'Brien align = elf_get_size(e, v, P_ALIGN); 7948fe960a1SDavid E. O'Brien fprintf(out, "\n"); 7958fe960a1SDavid E. O'Brien fprintf(out, "entry: %d\n", i); 7968fe960a1SDavid E. O'Brien fprintf(out, "\tp_type: %s\n", p_types[type & 0x7]); 79775ca4dcfSDavid E. O'Brien fprintf(out, "\tp_offset: %jd\n", (intmax_t)offset); 79875ca4dcfSDavid E. O'Brien fprintf(out, "\tp_vaddr: %#jx\n", (intmax_t)vaddr); 79975ca4dcfSDavid E. O'Brien fprintf(out, "\tp_paddr: %#jx\n", (intmax_t)paddr); 80075ca4dcfSDavid E. O'Brien fprintf(out, "\tp_filesz: %jd\n", (intmax_t)filesz); 80175ca4dcfSDavid E. O'Brien fprintf(out, "\tp_memsz: %jd\n", (intmax_t)memsz); 8028fe960a1SDavid E. O'Brien fprintf(out, "\tp_flags: %s\n", p_flags[flags]); 80375ca4dcfSDavid E. O'Brien fprintf(out, "\tp_align: %jd\n", (intmax_t)align); 8048fe960a1SDavid E. O'Brien } 8058fe960a1SDavid E. O'Brien } 8068fe960a1SDavid E. O'Brien 807bf70beceSEd Schouten static void 8080529c56bSDavid E. O'Brien elf_print_shdr(Elf32_Ehdr *e, void *sh) 8098fe960a1SDavid E. O'Brien { 8108fe960a1SDavid E. O'Brien u_int64_t shentsize; 8118fe960a1SDavid E. O'Brien u_int64_t shnum; 8128fe960a1SDavid E. O'Brien u_int64_t name; 8138fe960a1SDavid E. O'Brien u_int64_t type; 8148fe960a1SDavid E. O'Brien u_int64_t flags; 8158fe960a1SDavid E. O'Brien u_int64_t addr; 8168fe960a1SDavid E. O'Brien u_int64_t offset; 8178fe960a1SDavid E. O'Brien u_int64_t size; 81836210553SDavid E. O'Brien u_int64_t shlink; 8198fe960a1SDavid E. O'Brien u_int64_t info; 8208fe960a1SDavid E. O'Brien u_int64_t addralign; 8218fe960a1SDavid E. O'Brien u_int64_t entsize; 822b3854059SEd Maste u_int64_t machine; 8238fe960a1SDavid E. O'Brien void *v; 8248fe960a1SDavid E. O'Brien int i; 8258fe960a1SDavid E. O'Brien 826a589e596SMarcel Moolenaar if (sh == NULL) { 827a589e596SMarcel Moolenaar fprintf(out, "\nNo section headers\n"); 828a589e596SMarcel Moolenaar return; 829a589e596SMarcel Moolenaar } 830a589e596SMarcel Moolenaar 831b3854059SEd Maste machine = elf_get_quarter(e, e, E_MACHINE); 8328fe960a1SDavid E. O'Brien shentsize = elf_get_quarter(e, e, E_SHENTSIZE); 83397a9b0b1SEd Maste shnum = elf_get_shnum(e, sh); 8348fe960a1SDavid E. O'Brien fprintf(out, "\nsection header:\n"); 83536210553SDavid E. O'Brien for (i = 0; (u_int64_t)i < shnum; i++) { 83636210553SDavid E. O'Brien v = (char *)sh + i * shentsize; 8378fe960a1SDavid E. O'Brien name = elf_get_word(e, v, SH_NAME); 8388fe960a1SDavid E. O'Brien type = elf_get_word(e, v, SH_TYPE); 8398fe960a1SDavid E. O'Brien flags = elf_get_word(e, v, SH_FLAGS); 8408fe960a1SDavid E. O'Brien addr = elf_get_addr(e, v, SH_ADDR); 8418fe960a1SDavid E. O'Brien offset = elf_get_off(e, v, SH_OFFSET); 8428fe960a1SDavid E. O'Brien size = elf_get_size(e, v, SH_SIZE); 84336210553SDavid E. O'Brien shlink = elf_get_word(e, v, SH_LINK); 8448fe960a1SDavid E. O'Brien info = elf_get_word(e, v, SH_INFO); 8458fe960a1SDavid E. O'Brien addralign = elf_get_size(e, v, SH_ADDRALIGN); 8468fe960a1SDavid E. O'Brien entsize = elf_get_size(e, v, SH_ENTSIZE); 8478fe960a1SDavid E. O'Brien fprintf(out, "\n"); 8488fe960a1SDavid E. O'Brien fprintf(out, "entry: %d\n", i); 8498fe960a1SDavid E. O'Brien fprintf(out, "\tsh_name: %s\n", shstrtab + name); 850b3854059SEd Maste fprintf(out, "\tsh_type: %s\n", sh_types(machine, type)); 8518fe960a1SDavid E. O'Brien fprintf(out, "\tsh_flags: %s\n", sh_flags[flags & 0x7]); 85275ca4dcfSDavid E. O'Brien fprintf(out, "\tsh_addr: %#jx\n", addr); 85375ca4dcfSDavid E. O'Brien fprintf(out, "\tsh_offset: %jd\n", (intmax_t)offset); 85475ca4dcfSDavid E. O'Brien fprintf(out, "\tsh_size: %jd\n", (intmax_t)size); 85575ca4dcfSDavid E. O'Brien fprintf(out, "\tsh_link: %jd\n", (intmax_t)shlink); 85675ca4dcfSDavid E. O'Brien fprintf(out, "\tsh_info: %jd\n", (intmax_t)info); 85775ca4dcfSDavid E. O'Brien fprintf(out, "\tsh_addralign: %jd\n", (intmax_t)addralign); 85875ca4dcfSDavid E. O'Brien fprintf(out, "\tsh_entsize: %jd\n", (intmax_t)entsize); 8598fe960a1SDavid E. O'Brien } 8608fe960a1SDavid E. O'Brien } 8618fe960a1SDavid E. O'Brien 862bf70beceSEd Schouten static void 8630529c56bSDavid E. O'Brien elf_print_symtab(Elf32_Ehdr *e, void *sh, char *str) 8648fe960a1SDavid E. O'Brien { 8653a4b59a2SEd Maste u_int64_t machine; 8668fe960a1SDavid E. O'Brien u_int64_t offset; 8678fe960a1SDavid E. O'Brien u_int64_t entsize; 8688fe960a1SDavid E. O'Brien u_int64_t size; 8698fe960a1SDavid E. O'Brien u_int64_t name; 8708fe960a1SDavid E. O'Brien u_int64_t value; 8718fe960a1SDavid E. O'Brien u_int64_t info; 8728fe960a1SDavid E. O'Brien u_int64_t shndx; 8738fe960a1SDavid E. O'Brien void *st; 8748fe960a1SDavid E. O'Brien int len; 8758fe960a1SDavid E. O'Brien int i; 8768fe960a1SDavid E. O'Brien 8773a4b59a2SEd Maste machine = elf_get_quarter(e, e, E_MACHINE); 8788fe960a1SDavid E. O'Brien offset = elf_get_off(e, sh, SH_OFFSET); 8798fe960a1SDavid E. O'Brien entsize = elf_get_size(e, sh, SH_ENTSIZE); 8808fe960a1SDavid E. O'Brien size = elf_get_size(e, sh, SH_SIZE); 8818fe960a1SDavid E. O'Brien name = elf_get_word(e, sh, SH_NAME); 8828fe960a1SDavid E. O'Brien len = size / entsize; 8838fe960a1SDavid E. O'Brien fprintf(out, "\nsymbol table (%s):\n", shstrtab + name); 8848fe960a1SDavid E. O'Brien for (i = 0; i < len; i++) { 88536210553SDavid E. O'Brien st = (char *)e + offset + i * entsize; 8868fe960a1SDavid E. O'Brien name = elf_get_word(e, st, ST_NAME); 8878fe960a1SDavid E. O'Brien value = elf_get_addr(e, st, ST_VALUE); 8888fe960a1SDavid E. O'Brien size = elf_get_size(e, st, ST_SIZE); 8898fe960a1SDavid E. O'Brien info = elf_get_byte(e, st, ST_INFO); 8908fe960a1SDavid E. O'Brien shndx = elf_get_quarter(e, st, ST_SHNDX); 8918fe960a1SDavid E. O'Brien fprintf(out, "\n"); 8928fe960a1SDavid E. O'Brien fprintf(out, "entry: %d\n", i); 8938fe960a1SDavid E. O'Brien fprintf(out, "\tst_name: %s\n", str + name); 89475ca4dcfSDavid E. O'Brien fprintf(out, "\tst_value: %#jx\n", value); 89575ca4dcfSDavid E. O'Brien fprintf(out, "\tst_size: %jd\n", (intmax_t)size); 8968fe960a1SDavid E. O'Brien fprintf(out, "\tst_info: %s %s\n", 8973a4b59a2SEd Maste st_type(machine, ELF32_ST_TYPE(info)), 8988fe960a1SDavid E. O'Brien st_bindings[ELF32_ST_BIND(info)]); 89975ca4dcfSDavid E. O'Brien fprintf(out, "\tst_shndx: %jd\n", (intmax_t)shndx); 9008fe960a1SDavid E. O'Brien } 9018fe960a1SDavid E. O'Brien } 9028fe960a1SDavid E. O'Brien 903bf70beceSEd Schouten static void 9040529c56bSDavid E. O'Brien elf_print_dynamic(Elf32_Ehdr *e, void *sh) 9058fe960a1SDavid E. O'Brien { 9068fe960a1SDavid E. O'Brien u_int64_t offset; 9078fe960a1SDavid E. O'Brien u_int64_t entsize; 9088fe960a1SDavid E. O'Brien u_int64_t size; 9098fe960a1SDavid E. O'Brien int64_t tag; 9108fe960a1SDavid E. O'Brien u_int64_t ptr; 9118fe960a1SDavid E. O'Brien u_int64_t val; 9128fe960a1SDavid E. O'Brien void *d; 9138fe960a1SDavid E. O'Brien int i; 9148fe960a1SDavid E. O'Brien 9158fe960a1SDavid E. O'Brien offset = elf_get_off(e, sh, SH_OFFSET); 9168fe960a1SDavid E. O'Brien entsize = elf_get_size(e, sh, SH_ENTSIZE); 9178fe960a1SDavid E. O'Brien size = elf_get_size(e, sh, SH_SIZE); 9188fe960a1SDavid E. O'Brien fprintf(out, "\ndynamic:\n"); 91936210553SDavid E. O'Brien for (i = 0; (u_int64_t)i < size / entsize; i++) { 92036210553SDavid E. O'Brien d = (char *)e + offset + i * entsize; 9218fe960a1SDavid E. O'Brien tag = elf_get_size(e, d, D_TAG); 9228fe960a1SDavid E. O'Brien ptr = elf_get_size(e, d, D_PTR); 9238fe960a1SDavid E. O'Brien val = elf_get_addr(e, d, D_VAL); 9248fe960a1SDavid E. O'Brien fprintf(out, "\n"); 9258fe960a1SDavid E. O'Brien fprintf(out, "entry: %d\n", i); 9263df3bb12SDavid E. O'Brien fprintf(out, "\td_tag: %s\n", d_tags(tag)); 9278fe960a1SDavid E. O'Brien switch (tag) { 9288fe960a1SDavid E. O'Brien case DT_NEEDED: 9298fe960a1SDavid E. O'Brien case DT_SONAME: 9308fe960a1SDavid E. O'Brien case DT_RPATH: 9318fe960a1SDavid E. O'Brien fprintf(out, "\td_val: %s\n", dynstr + val); 9328fe960a1SDavid E. O'Brien break; 9338fe960a1SDavid E. O'Brien case DT_PLTRELSZ: 9348fe960a1SDavid E. O'Brien case DT_RELA: 9358fe960a1SDavid E. O'Brien case DT_RELASZ: 9368fe960a1SDavid E. O'Brien case DT_RELAENT: 9378fe960a1SDavid E. O'Brien case DT_STRSZ: 9388fe960a1SDavid E. O'Brien case DT_SYMENT: 9398fe960a1SDavid E. O'Brien case DT_RELSZ: 9408fe960a1SDavid E. O'Brien case DT_RELENT: 9418fe960a1SDavid E. O'Brien case DT_PLTREL: 94275ca4dcfSDavid E. O'Brien fprintf(out, "\td_val: %jd\n", (intmax_t)val); 9438fe960a1SDavid E. O'Brien break; 9448fe960a1SDavid E. O'Brien case DT_PLTGOT: 9458fe960a1SDavid E. O'Brien case DT_HASH: 9468fe960a1SDavid E. O'Brien case DT_STRTAB: 9478fe960a1SDavid E. O'Brien case DT_SYMTAB: 9488fe960a1SDavid E. O'Brien case DT_INIT: 9498fe960a1SDavid E. O'Brien case DT_FINI: 9508fe960a1SDavid E. O'Brien case DT_REL: 9518fe960a1SDavid E. O'Brien case DT_JMPREL: 95275ca4dcfSDavid E. O'Brien fprintf(out, "\td_ptr: %#jx\n", ptr); 9538fe960a1SDavid E. O'Brien break; 9548fe960a1SDavid E. O'Brien case DT_NULL: 9558fe960a1SDavid E. O'Brien case DT_SYMBOLIC: 9568fe960a1SDavid E. O'Brien case DT_DEBUG: 9578fe960a1SDavid E. O'Brien case DT_TEXTREL: 9588fe960a1SDavid E. O'Brien break; 9598fe960a1SDavid E. O'Brien } 9608fe960a1SDavid E. O'Brien } 9618fe960a1SDavid E. O'Brien } 9628fe960a1SDavid E. O'Brien 963bf70beceSEd Schouten static void 9640529c56bSDavid E. O'Brien elf_print_rela(Elf32_Ehdr *e, void *sh) 9658fe960a1SDavid E. O'Brien { 9668fe960a1SDavid E. O'Brien u_int64_t offset; 9678fe960a1SDavid E. O'Brien u_int64_t entsize; 9688fe960a1SDavid E. O'Brien u_int64_t size; 9698fe960a1SDavid E. O'Brien u_int64_t name; 9708fe960a1SDavid E. O'Brien u_int64_t info; 9718fe960a1SDavid E. O'Brien int64_t addend; 9728fe960a1SDavid E. O'Brien void *ra; 9738fe960a1SDavid E. O'Brien void *v; 9748fe960a1SDavid E. O'Brien int i; 9758fe960a1SDavid E. O'Brien 9768fe960a1SDavid E. O'Brien offset = elf_get_off(e, sh, SH_OFFSET); 9778fe960a1SDavid E. O'Brien entsize = elf_get_size(e, sh, SH_ENTSIZE); 9788fe960a1SDavid E. O'Brien size = elf_get_size(e, sh, SH_SIZE); 9798fe960a1SDavid E. O'Brien name = elf_get_word(e, sh, SH_NAME); 98036210553SDavid E. O'Brien v = (char *)e + offset; 9818fe960a1SDavid E. O'Brien fprintf(out, "\nrelocation with addend (%s):\n", shstrtab + name); 98236210553SDavid E. O'Brien for (i = 0; (u_int64_t)i < size / entsize; i++) { 98336210553SDavid E. O'Brien ra = (char *)v + i * entsize; 9848fe960a1SDavid E. O'Brien offset = elf_get_addr(e, ra, RA_OFFSET); 9858fe960a1SDavid E. O'Brien info = elf_get_word(e, ra, RA_INFO); 9868fe960a1SDavid E. O'Brien addend = elf_get_off(e, ra, RA_ADDEND); 9878fe960a1SDavid E. O'Brien fprintf(out, "\n"); 9888fe960a1SDavid E. O'Brien fprintf(out, "entry: %d\n", i); 98975ca4dcfSDavid E. O'Brien fprintf(out, "\tr_offset: %#jx\n", offset); 99075ca4dcfSDavid E. O'Brien fprintf(out, "\tr_info: %jd\n", (intmax_t)info); 99175ca4dcfSDavid E. O'Brien fprintf(out, "\tr_addend: %jd\n", (intmax_t)addend); 9928fe960a1SDavid E. O'Brien } 9938fe960a1SDavid E. O'Brien } 9948fe960a1SDavid E. O'Brien 995bf70beceSEd Schouten static void 9960529c56bSDavid E. O'Brien elf_print_rel(Elf32_Ehdr *e, void *sh) 9978fe960a1SDavid E. O'Brien { 9988fe960a1SDavid E. O'Brien u_int64_t offset; 9998fe960a1SDavid E. O'Brien u_int64_t entsize; 10008fe960a1SDavid E. O'Brien u_int64_t size; 10018fe960a1SDavid E. O'Brien u_int64_t name; 10028fe960a1SDavid E. O'Brien u_int64_t info; 10038fe960a1SDavid E. O'Brien void *r; 10048fe960a1SDavid E. O'Brien void *v; 10058fe960a1SDavid E. O'Brien int i; 10068fe960a1SDavid E. O'Brien 10078fe960a1SDavid E. O'Brien offset = elf_get_off(e, sh, SH_OFFSET); 10088fe960a1SDavid E. O'Brien entsize = elf_get_size(e, sh, SH_ENTSIZE); 10098fe960a1SDavid E. O'Brien size = elf_get_size(e, sh, SH_SIZE); 10108fe960a1SDavid E. O'Brien name = elf_get_word(e, sh, SH_NAME); 101136210553SDavid E. O'Brien v = (char *)e + offset; 10128fe960a1SDavid E. O'Brien fprintf(out, "\nrelocation (%s):\n", shstrtab + name); 101336210553SDavid E. O'Brien for (i = 0; (u_int64_t)i < size / entsize; i++) { 101436210553SDavid E. O'Brien r = (char *)v + i * entsize; 10158fe960a1SDavid E. O'Brien offset = elf_get_addr(e, r, R_OFFSET); 10168fe960a1SDavid E. O'Brien info = elf_get_word(e, r, R_INFO); 10178fe960a1SDavid E. O'Brien fprintf(out, "\n"); 10188fe960a1SDavid E. O'Brien fprintf(out, "entry: %d\n", i); 101975ca4dcfSDavid E. O'Brien fprintf(out, "\tr_offset: %#jx\n", offset); 102075ca4dcfSDavid E. O'Brien fprintf(out, "\tr_info: %jd\n", (intmax_t)info); 10218fe960a1SDavid E. O'Brien } 10228fe960a1SDavid E. O'Brien } 10238fe960a1SDavid E. O'Brien 1024bf70beceSEd Schouten static void 10250529c56bSDavid E. O'Brien elf_print_interp(Elf32_Ehdr *e, void *p) 10268fe960a1SDavid E. O'Brien { 10278fe960a1SDavid E. O'Brien u_int64_t offset; 10288fe960a1SDavid E. O'Brien char *s; 10298fe960a1SDavid E. O'Brien 10308fe960a1SDavid E. O'Brien offset = elf_get_off(e, p, P_OFFSET); 10310529c56bSDavid E. O'Brien s = (char *)e + offset; 10328fe960a1SDavid E. O'Brien fprintf(out, "\ninterp:\n"); 10338fe960a1SDavid E. O'Brien fprintf(out, "\t%s\n", s); 10348fe960a1SDavid E. O'Brien } 10358fe960a1SDavid E. O'Brien 1036bf70beceSEd Schouten static void 10370529c56bSDavid E. O'Brien elf_print_got(Elf32_Ehdr *e, void *sh) 10388fe960a1SDavid E. O'Brien { 10398fe960a1SDavid E. O'Brien u_int64_t offset; 10408fe960a1SDavid E. O'Brien u_int64_t addralign; 10418fe960a1SDavid E. O'Brien u_int64_t size; 10428fe960a1SDavid E. O'Brien u_int64_t addr; 10438fe960a1SDavid E. O'Brien void *v; 10448fe960a1SDavid E. O'Brien int i; 10458fe960a1SDavid E. O'Brien 10468fe960a1SDavid E. O'Brien offset = elf_get_off(e, sh, SH_OFFSET); 10478fe960a1SDavid E. O'Brien addralign = elf_get_size(e, sh, SH_ADDRALIGN); 10488fe960a1SDavid E. O'Brien size = elf_get_size(e, sh, SH_SIZE); 104936210553SDavid E. O'Brien v = (char *)e + offset; 10508fe960a1SDavid E. O'Brien fprintf(out, "\nglobal offset table:\n"); 105136210553SDavid E. O'Brien for (i = 0; (u_int64_t)i < size / addralign; i++) { 105236210553SDavid E. O'Brien addr = elf_get_addr(e, (char *)v + i * addralign, 0); 10538fe960a1SDavid E. O'Brien fprintf(out, "\n"); 10548fe960a1SDavid E. O'Brien fprintf(out, "entry: %d\n", i); 105575ca4dcfSDavid E. O'Brien fprintf(out, "\t%#jx\n", addr); 10568fe960a1SDavid E. O'Brien } 10578fe960a1SDavid E. O'Brien } 10588fe960a1SDavid E. O'Brien 1059bf70beceSEd Schouten static void 106036210553SDavid E. O'Brien elf_print_hash(Elf32_Ehdr *e __unused, void *sh __unused) 10618fe960a1SDavid E. O'Brien { 10628fe960a1SDavid E. O'Brien } 10638fe960a1SDavid E. O'Brien 1064bf70beceSEd Schouten static void 10650529c56bSDavid E. O'Brien elf_print_note(Elf32_Ehdr *e, void *sh) 10668fe960a1SDavid E. O'Brien { 10678fe960a1SDavid E. O'Brien u_int64_t offset; 10688fe960a1SDavid E. O'Brien u_int64_t size; 10698fe960a1SDavid E. O'Brien u_int64_t name; 10708fe960a1SDavid E. O'Brien u_int32_t namesz; 10718fe960a1SDavid E. O'Brien u_int32_t descsz; 10728fe960a1SDavid E. O'Brien u_int32_t desc; 10738ae9aa27SEd Maste u_int32_t type; 107436210553SDavid E. O'Brien char *n, *s; 10758ae9aa27SEd Maste const char *nt_type; 10768fe960a1SDavid E. O'Brien 10778fe960a1SDavid E. O'Brien offset = elf_get_off(e, sh, SH_OFFSET); 10788fe960a1SDavid E. O'Brien size = elf_get_size(e, sh, SH_SIZE); 10798fe960a1SDavid E. O'Brien name = elf_get_word(e, sh, SH_NAME); 108036210553SDavid E. O'Brien n = (char *)e + offset; 10818fe960a1SDavid E. O'Brien fprintf(out, "\nnote (%s):\n", shstrtab + name); 108236210553SDavid E. O'Brien while (n < ((char *)e + offset + size)) { 10838fe960a1SDavid E. O'Brien namesz = elf_get_word(e, n, N_NAMESZ); 10848fe960a1SDavid E. O'Brien descsz = elf_get_word(e, n, N_DESCSZ); 10852bc7b024SEd Maste type = elf_get_word(e, n, N_TYPE); 1086*97d368d6SEd Maste if (type < nitems(nt_types) && nt_types[type] != NULL) 10872bc7b024SEd Maste nt_type = nt_types[type]; 10882bc7b024SEd Maste else 10892bc7b024SEd Maste nt_type = "Unknown type"; 10908fe960a1SDavid E. O'Brien s = n + sizeof(Elf_Note); 10918fe960a1SDavid E. O'Brien desc = elf_get_word(e, n + sizeof(Elf_Note) + namesz, 0); 10922bc7b024SEd Maste fprintf(out, "\t%s %d (%s)\n", s, desc, nt_type); 10938fe960a1SDavid E. O'Brien n += sizeof(Elf_Note) + namesz + descsz; 10948fe960a1SDavid E. O'Brien } 10958fe960a1SDavid E. O'Brien } 10968fe960a1SDavid E. O'Brien 1097bf70beceSEd Schouten static u_int64_t 10988fe960a1SDavid E. O'Brien elf_get_byte(Elf32_Ehdr *e, void *base, elf_member_t member) 10998fe960a1SDavid E. O'Brien { 11008fe960a1SDavid E. O'Brien u_int64_t val; 11018fe960a1SDavid E. O'Brien 11028fe960a1SDavid E. O'Brien val = 0; 11038fe960a1SDavid E. O'Brien switch (e->e_ident[EI_CLASS]) { 11048fe960a1SDavid E. O'Brien case ELFCLASS32: 1105ead61635SMarcel Moolenaar val = ((uint8_t *)base)[elf32_offsets[member]]; 11068fe960a1SDavid E. O'Brien break; 11078fe960a1SDavid E. O'Brien case ELFCLASS64: 1108ead61635SMarcel Moolenaar val = ((uint8_t *)base)[elf64_offsets[member]]; 11098fe960a1SDavid E. O'Brien break; 11108fe960a1SDavid E. O'Brien case ELFCLASSNONE: 11118fe960a1SDavid E. O'Brien errx(1, "invalid class"); 11128fe960a1SDavid E. O'Brien } 11138fe960a1SDavid E. O'Brien 11148fe960a1SDavid E. O'Brien return val; 11158fe960a1SDavid E. O'Brien } 11168fe960a1SDavid E. O'Brien 1117bf70beceSEd Schouten static u_int64_t 11188fe960a1SDavid E. O'Brien elf_get_quarter(Elf32_Ehdr *e, void *base, elf_member_t member) 11198fe960a1SDavid E. O'Brien { 11208fe960a1SDavid E. O'Brien u_int64_t val; 11218fe960a1SDavid E. O'Brien 11228fe960a1SDavid E. O'Brien val = 0; 11238fe960a1SDavid E. O'Brien switch (e->e_ident[EI_CLASS]) { 11248fe960a1SDavid E. O'Brien case ELFCLASS32: 1125ac2ded1dSMarcel Moolenaar base = (char *)base + elf32_offsets[member]; 11268fe960a1SDavid E. O'Brien switch (e->e_ident[EI_DATA]) { 11278fe960a1SDavid E. O'Brien case ELFDATA2MSB: 1128ac2ded1dSMarcel Moolenaar val = be16dec(base); 11298fe960a1SDavid E. O'Brien break; 11308fe960a1SDavid E. O'Brien case ELFDATA2LSB: 1131ac2ded1dSMarcel Moolenaar val = le16dec(base); 11328fe960a1SDavid E. O'Brien break; 11338fe960a1SDavid E. O'Brien case ELFDATANONE: 11348fe960a1SDavid E. O'Brien errx(1, "invalid data format"); 11358fe960a1SDavid E. O'Brien } 11368fe960a1SDavid E. O'Brien break; 11378fe960a1SDavid E. O'Brien case ELFCLASS64: 1138ac2ded1dSMarcel Moolenaar base = (char *)base + elf64_offsets[member]; 11398fe960a1SDavid E. O'Brien switch (e->e_ident[EI_DATA]) { 11408fe960a1SDavid E. O'Brien case ELFDATA2MSB: 1141ac2ded1dSMarcel Moolenaar val = be16dec(base); 11428fe960a1SDavid E. O'Brien break; 11438fe960a1SDavid E. O'Brien case ELFDATA2LSB: 1144ac2ded1dSMarcel Moolenaar val = le16dec(base); 11458fe960a1SDavid E. O'Brien break; 11468fe960a1SDavid E. O'Brien case ELFDATANONE: 11478fe960a1SDavid E. O'Brien errx(1, "invalid data format"); 11488fe960a1SDavid E. O'Brien } 11498fe960a1SDavid E. O'Brien break; 11508fe960a1SDavid E. O'Brien case ELFCLASSNONE: 11518fe960a1SDavid E. O'Brien errx(1, "invalid class"); 11528fe960a1SDavid E. O'Brien } 11538fe960a1SDavid E. O'Brien 11548fe960a1SDavid E. O'Brien return val; 11558fe960a1SDavid E. O'Brien } 11568fe960a1SDavid E. O'Brien 1157bf70beceSEd Schouten #if 0 1158bf70beceSEd Schouten static u_int64_t 11598fe960a1SDavid E. O'Brien elf_get_half(Elf32_Ehdr *e, void *base, elf_member_t member) 11608fe960a1SDavid E. O'Brien { 11618fe960a1SDavid E. O'Brien u_int64_t val; 11628fe960a1SDavid E. O'Brien 11638fe960a1SDavid E. O'Brien val = 0; 11648fe960a1SDavid E. O'Brien switch (e->e_ident[EI_CLASS]) { 11658fe960a1SDavid E. O'Brien case ELFCLASS32: 1166ac2ded1dSMarcel Moolenaar base = (char *)base + elf32_offsets[member]; 11678fe960a1SDavid E. O'Brien switch (e->e_ident[EI_DATA]) { 11688fe960a1SDavid E. O'Brien case ELFDATA2MSB: 1169ac2ded1dSMarcel Moolenaar val = be16dec(base); 11708fe960a1SDavid E. O'Brien break; 11718fe960a1SDavid E. O'Brien case ELFDATA2LSB: 1172ac2ded1dSMarcel Moolenaar val = le16dec(base); 11738fe960a1SDavid E. O'Brien break; 11748fe960a1SDavid E. O'Brien case ELFDATANONE: 11758fe960a1SDavid E. O'Brien errx(1, "invalid data format"); 11768fe960a1SDavid E. O'Brien } 11778fe960a1SDavid E. O'Brien break; 11788fe960a1SDavid E. O'Brien case ELFCLASS64: 1179ac2ded1dSMarcel Moolenaar base = (char *)base + elf64_offsets[member]; 11808fe960a1SDavid E. O'Brien switch (e->e_ident[EI_DATA]) { 11818fe960a1SDavid E. O'Brien case ELFDATA2MSB: 1182ac2ded1dSMarcel Moolenaar val = be32dec(base); 11838fe960a1SDavid E. O'Brien break; 11848fe960a1SDavid E. O'Brien case ELFDATA2LSB: 1185ac2ded1dSMarcel Moolenaar val = le32dec(base); 11868fe960a1SDavid E. O'Brien break; 11878fe960a1SDavid E. O'Brien case ELFDATANONE: 11888fe960a1SDavid E. O'Brien errx(1, "invalid data format"); 11898fe960a1SDavid E. O'Brien } 11908fe960a1SDavid E. O'Brien break; 11918fe960a1SDavid E. O'Brien case ELFCLASSNONE: 11928fe960a1SDavid E. O'Brien errx(1, "invalid class"); 11938fe960a1SDavid E. O'Brien } 11948fe960a1SDavid E. O'Brien 11958fe960a1SDavid E. O'Brien return val; 11968fe960a1SDavid E. O'Brien } 1197bf70beceSEd Schouten #endif 11988fe960a1SDavid E. O'Brien 1199bf70beceSEd Schouten static u_int64_t 12008fe960a1SDavid E. O'Brien elf_get_word(Elf32_Ehdr *e, void *base, elf_member_t member) 12018fe960a1SDavid E. O'Brien { 12028fe960a1SDavid E. O'Brien u_int64_t val; 12038fe960a1SDavid E. O'Brien 12048fe960a1SDavid E. O'Brien val = 0; 12058fe960a1SDavid E. O'Brien switch (e->e_ident[EI_CLASS]) { 12068fe960a1SDavid E. O'Brien case ELFCLASS32: 1207ac2ded1dSMarcel Moolenaar base = (char *)base + elf32_offsets[member]; 12088fe960a1SDavid E. O'Brien switch (e->e_ident[EI_DATA]) { 12098fe960a1SDavid E. O'Brien case ELFDATA2MSB: 1210ac2ded1dSMarcel Moolenaar val = be32dec(base); 12118fe960a1SDavid E. O'Brien break; 12128fe960a1SDavid E. O'Brien case ELFDATA2LSB: 1213ac2ded1dSMarcel Moolenaar val = le32dec(base); 12148fe960a1SDavid E. O'Brien break; 12158fe960a1SDavid E. O'Brien case ELFDATANONE: 12168fe960a1SDavid E. O'Brien errx(1, "invalid data format"); 12178fe960a1SDavid E. O'Brien } 12188fe960a1SDavid E. O'Brien break; 12198fe960a1SDavid E. O'Brien case ELFCLASS64: 1220ac2ded1dSMarcel Moolenaar base = (char *)base + elf64_offsets[member]; 12218fe960a1SDavid E. O'Brien switch (e->e_ident[EI_DATA]) { 12228fe960a1SDavid E. O'Brien case ELFDATA2MSB: 1223ac2ded1dSMarcel Moolenaar val = be32dec(base); 12248fe960a1SDavid E. O'Brien break; 12258fe960a1SDavid E. O'Brien case ELFDATA2LSB: 1226ac2ded1dSMarcel Moolenaar val = le32dec(base); 12278fe960a1SDavid E. O'Brien break; 12288fe960a1SDavid E. O'Brien case ELFDATANONE: 12298fe960a1SDavid E. O'Brien errx(1, "invalid data format"); 12308fe960a1SDavid E. O'Brien } 12318fe960a1SDavid E. O'Brien break; 12328fe960a1SDavid E. O'Brien case ELFCLASSNONE: 12338fe960a1SDavid E. O'Brien errx(1, "invalid class"); 12348fe960a1SDavid E. O'Brien } 12358fe960a1SDavid E. O'Brien 12368fe960a1SDavid E. O'Brien return val; 12378fe960a1SDavid E. O'Brien } 12388fe960a1SDavid E. O'Brien 1239bf70beceSEd Schouten static u_int64_t 12408fe960a1SDavid E. O'Brien elf_get_quad(Elf32_Ehdr *e, void *base, elf_member_t member) 12418fe960a1SDavid E. O'Brien { 12428fe960a1SDavid E. O'Brien u_int64_t val; 12438fe960a1SDavid E. O'Brien 12448fe960a1SDavid E. O'Brien val = 0; 12458fe960a1SDavid E. O'Brien switch (e->e_ident[EI_CLASS]) { 12468fe960a1SDavid E. O'Brien case ELFCLASS32: 1247ac2ded1dSMarcel Moolenaar base = (char *)base + elf32_offsets[member]; 12488fe960a1SDavid E. O'Brien switch (e->e_ident[EI_DATA]) { 12498fe960a1SDavid E. O'Brien case ELFDATA2MSB: 1250ac2ded1dSMarcel Moolenaar val = be32dec(base); 12518fe960a1SDavid E. O'Brien break; 12528fe960a1SDavid E. O'Brien case ELFDATA2LSB: 1253ac2ded1dSMarcel Moolenaar val = le32dec(base); 12548fe960a1SDavid E. O'Brien break; 12558fe960a1SDavid E. O'Brien case ELFDATANONE: 12568fe960a1SDavid E. O'Brien errx(1, "invalid data format"); 12578fe960a1SDavid E. O'Brien } 12588fe960a1SDavid E. O'Brien break; 12598fe960a1SDavid E. O'Brien case ELFCLASS64: 1260ac2ded1dSMarcel Moolenaar base = (char *)base + elf64_offsets[member]; 12618fe960a1SDavid E. O'Brien switch (e->e_ident[EI_DATA]) { 12628fe960a1SDavid E. O'Brien case ELFDATA2MSB: 1263ac2ded1dSMarcel Moolenaar val = be64dec(base); 12648fe960a1SDavid E. O'Brien break; 12658fe960a1SDavid E. O'Brien case ELFDATA2LSB: 1266ac2ded1dSMarcel Moolenaar val = le64dec(base); 12678fe960a1SDavid E. O'Brien break; 12688fe960a1SDavid E. O'Brien case ELFDATANONE: 12698fe960a1SDavid E. O'Brien errx(1, "invalid data format"); 12708fe960a1SDavid E. O'Brien } 12718fe960a1SDavid E. O'Brien break; 12728fe960a1SDavid E. O'Brien case ELFCLASSNONE: 12738fe960a1SDavid E. O'Brien errx(1, "invalid class"); 12748fe960a1SDavid E. O'Brien } 12758fe960a1SDavid E. O'Brien 12768fe960a1SDavid E. O'Brien return val; 12778fe960a1SDavid E. O'Brien } 12788fe960a1SDavid E. O'Brien 1279bf70beceSEd Schouten static void 12808fe960a1SDavid E. O'Brien usage(void) 12818fe960a1SDavid E. O'Brien { 1282c35530f4SBrooks Davis fprintf(stderr, 1283c35530f4SBrooks Davis "usage: elfdump -a | -E | -cdeGhinprs [-w file] file\n"); 12848fe960a1SDavid E. O'Brien exit(1); 12858fe960a1SDavid E. O'Brien } 1286