1345 lines
54 KiB
C++
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
Copyright (C) 2001-present by Serge Lamikhov-Center
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
*/
#ifndef ELFIO_DUMP_HPP
#define ELFIO_DUMP_HPP
#include <algorithm>
#include <string>
#include <iomanip>
#include <elfio/elfio.hpp>
namespace ELFIO {
static const struct class_table_t
{
const char key;
const char* str;
} class_table[] = {
{ ELFCLASS32, "ELF32" },
{ ELFCLASS64, "ELF64" },
};
static const struct endian_table_t
{
const char key;
const char* str;
} endian_table[] = {
{ ELFDATANONE, "None" },
{ ELFDATA2LSB, "Little endian" },
{ ELFDATA2MSB, "Big endian" },
};
static const struct os_abi_table_t
{
const unsigned char key;
const char* str;
} os_abi_table[] = {
{ ELFOSABI_NONE, "UNIX System V" },
{ ELFOSABI_HPUX, "Hewlett-Packard HP-UX" },
{ ELFOSABI_NETBSD, "NetBSD" },
{ ELFOSABI_LINUX, "Linux" },
{ ELFOSABI_HURD, "GNU Hurd" },
{ ELFOSABI_SOLARIS, "Sun Solaris" },
{ ELFOSABI_AIX, "AIX" },
{ ELFOSABI_IRIX, "IRIX" },
{ ELFOSABI_FREEBSD, "FreeBSD" },
{ ELFOSABI_TRU64, "Compaq TRU64 UNIX" },
{ ELFOSABI_MODESTO, "Novell Modesto" },
{ ELFOSABI_OPENBSD, "Open BSD" },
{ ELFOSABI_OPENVMS, "Open VMS" },
{ ELFOSABI_NSK, "Hewlett-Packard Non-Stop Kernel" },
{ ELFOSABI_AROS, "Amiga Research OS" },
{ ELFOSABI_FENIXOS, "FenixOS" },
{ ELFOSABI_NUXI, "Nuxi CloudABI" },
{ ELFOSABI_OPENVOS, "OpenVOS" },
{ ELFOSABI_ARM, "ARM" },
{ ELFOSABI_STANDALONE, "Standalone (embedded)" },
};
static const struct version_table_t
{
const Elf64_Word key;
const char* str;
} version_table[] = {
{ EV_NONE, "None" },
{ EV_CURRENT, "Current" },
};
static const struct type_table_t
{
const Elf32_Half key;
const char* str;
} type_table[] = {
{ ET_NONE, "No file type" }, { ET_REL, "Relocatable file" },
{ ET_EXEC, "Executable file" }, { ET_DYN, "Shared object file" },
{ ET_CORE, "Core file" },
};
static const struct machine_table_t
{
const Elf64_Half key;
const char* str;
} machine_table[] = {
{ EM_NONE, "No machine" },
{ EM_M32, "AT&T WE 32100" },
{ EM_SPARC, "SUN SPARC" },
{ EM_386, "Intel 80386" },
{ EM_68K, "Motorola m68k family" },
{ EM_88K, "Motorola m88k family" },
{ EM_486, "Intel 80486// Reserved for future use" },
{ EM_860, "Intel 80860" },
{ EM_MIPS, "MIPS R3000 (officially, big-endian only)" },
{ EM_S370, "IBM System/370" },
{ EM_MIPS_RS3_LE,
"MIPS R3000 little-endian (Oct 4 1999 Draft) Deprecated" },
{ EM_res011, "Reserved" },
{ EM_res012, "Reserved" },
{ EM_res013, "Reserved" },
{ EM_res014, "Reserved" },
{ EM_PARISC, "HPPA" },
{ EM_res016, "Reserved" },
{ EM_VPP550, "Fujitsu VPP500" },
{ EM_SPARC32PLUS, "Sun's v8plus" },
{ EM_960, "Intel 80960" },
{ EM_PPC, "PowerPC" },
{ EM_PPC64, "64-bit PowerPC" },
{ EM_S390, "IBM S/390" },
{ EM_SPU, "Sony/Toshiba/IBM SPU" },
{ EM_res024, "Reserved" },
{ EM_res025, "Reserved" },
{ EM_res026, "Reserved" },
{ EM_res027, "Reserved" },
{ EM_res028, "Reserved" },
{ EM_res029, "Reserved" },
{ EM_res030, "Reserved" },
{ EM_res031, "Reserved" },
{ EM_res032, "Reserved" },
{ EM_res033, "Reserved" },
{ EM_res034, "Reserved" },
{ EM_res035, "Reserved" },
{ EM_V800, "NEC V800 series" },
{ EM_FR20, "Fujitsu FR20" },
{ EM_RH32, "TRW RH32" },
{ EM_MCORE, "Motorola M*Core // May also be taken by Fujitsu MMA" },
{ EM_RCE, "Old name for MCore" },
{ EM_ARM, "ARM" },
{ EM_OLD_ALPHA, "Digital Alpha" },
{ EM_SH, "Renesas (formerly Hitachi) / SuperH SH" },
{ EM_SPARCV9, "SPARC v9 64-bit" },
{ EM_TRICORE, "Siemens Tricore embedded processor" },
{ EM_ARC, "ARC Cores" },
{ EM_H8_300, "Renesas (formerly Hitachi) H8/300" },
{ EM_H8_300H, "Renesas (formerly Hitachi) H8/300H" },
{ EM_H8S, "Renesas (formerly Hitachi) H8S" },
{ EM_H8_500, "Renesas (formerly Hitachi) H8/500" },
{ EM_IA_64, "Intel IA-64 Processor" },
{ EM_MIPS_X, "Stanford MIPS-X" },
{ EM_COLDFIRE, "Motorola Coldfire" },
{ EM_68HC12, "Motorola M68HC12" },
{ EM_MMA, "Fujitsu Multimedia Accelerator" },
{ EM_PCP, "Siemens PCP" },
{ EM_NCPU, "Sony nCPU embedded RISC processor" },
{ EM_NDR1, "Denso NDR1 microprocesspr" },
{ EM_STARCORE, "Motorola Star*Core processor" },
{ EM_ME16, "Toyota ME16 processor" },
{ EM_ST100, "STMicroelectronics ST100 processor" },
{ EM_TINYJ, "Advanced Logic Corp. TinyJ embedded processor" },
{ EM_X86_64, "Advanced Micro Devices X86-64 processor" },
{ EM_PDSP, "Sony DSP Processor" },
{ EM_PDP10, "Digital Equipment Corp. PDP-10" },
{ EM_PDP11, "Digital Equipment Corp. PDP-11" },
{ EM_FX66, "Siemens FX66 microcontroller" },
{ EM_ST9PLUS, "STMicroelectronics ST9+ 8/16 bit microcontroller" },
{ EM_ST7, "STMicroelectronics ST7 8-bit microcontroller" },
{ EM_68HC16, "Motorola MC68HC16 Microcontroller" },
{ EM_68HC11, "Motorola MC68HC11 Microcontroller" },
{ EM_68HC08, "Motorola MC68HC08 Microcontroller" },
{ EM_68HC05, "Motorola MC68HC05 Microcontroller" },
{ EM_SVX, "Silicon Graphics SVx" },
{ EM_ST19, "STMicroelectronics ST19 8-bit cpu" },
{ EM_VAX, "Digital VAX" },
{ EM_CRIS, "Axis Communications 32-bit embedded processor" },
{ EM_JAVELIN, "Infineon Technologies 32-bit embedded cpu" },
{ EM_FIREPATH, "Element 14 64-bit DSP processor" },
{ EM_ZSP, "LSI Logic's 16-bit DSP processor" },
{ EM_MMIX, "Donald Knuth's educational 64-bit processor" },
{ EM_HUANY, "Harvard's machine-independent format" },
{ EM_PRISM, "SiTera Prism" },
{ EM_AVR, "Atmel AVR 8-bit microcontroller" },
{ EM_FR30, "Fujitsu FR30" },
{ EM_D10V, "Mitsubishi D10V" },
{ EM_D30V, "Mitsubishi D30V" },
{ EM_V850, "NEC v850" },
{ EM_M32R, "Renesas M32R (formerly Mitsubishi M32R)" },
{ EM_MN10300, "Matsushita MN10300" },
{ EM_MN10200, "Matsushita MN10200" },
{ EM_PJ, "picoJava" },
{ EM_OPENRISC, "OpenRISC 32-bit embedded processor" },
{ EM_ARC_A5, "ARC Cores Tangent-A5" },
{ EM_XTENSA, "Tensilica Xtensa Architecture" },
{ EM_VIDEOCORE, "Alphamosaic VideoCore processor" },
{ EM_TMM_GPP, "Thompson Multimedia General Purpose Processor" },
{ EM_NS32K, "National Semiconductor 32000 series" },
{ EM_TPC, "Tenor Network TPC processor" },
{ EM_SNP1K, "Trebia SNP 1000 processor" },
{ EM_ST200, "STMicroelectronics ST200 microcontroller" },
{ EM_IP2K, "Ubicom IP2022 micro controller" },
{ EM_MAX, "MAX Processor" },
{ EM_CR, "National Semiconductor CompactRISC" },
{ EM_F2MC16, "Fujitsu F2MC16" },
{ EM_MSP430, "TI msp430 micro controller" },
{ EM_BLACKFIN, "ADI Blackfin" },
{ EM_SE_C33, "S1C33 Family of Seiko Epson processors" },
{ EM_SEP, "Sharp embedded microprocessor" },
{ EM_ARCA, "Arca RISC Microprocessor" },
{ EM_UNICORE, "Microprocessor series from PKU-Unity Ltd. and MPRC of "
"Peking University" },
{ EM_EXCESS, "eXcess: 16/32/64-bit configurable embedded CPU" },
{ EM_DXP, "Icera Semiconductor Inc. Deep Execution Processor" },
{ EM_ALTERA_NIOS2, "Altera Nios II soft-core processor" },
{ EM_CRX, "National Semiconductor CRX" },
{ EM_XGATE, "Motorola XGATE embedded processor" },
{ EM_C166, "Infineon C16x/XC16x processor" },
{ EM_M16C, "Renesas M16C series microprocessors" },
{ EM_DSPIC30F, "Microchip Technology dsPIC30F Digital Signal Controller" },
{ EM_CE, "Freescale Communication Engine RISC core" },
{ EM_M32C, "Renesas M32C series microprocessors" },
{ EM_res121, "Reserved" },
{ EM_res122, "Reserved" },
{ EM_res123, "Reserved" },
{ EM_res124, "Reserved" },
{ EM_res125, "Reserved" },
{ EM_res126, "Reserved" },
{ EM_res127, "Reserved" },
{ EM_res128, "Reserved" },
{ EM_res129, "Reserved" },
{ EM_res130, "Reserved" },
{ EM_TSK3000, "Altium TSK3000 core" },
{ EM_RS08, "Freescale RS08 embedded processor" },
{ EM_res133, "Reserved" },
{ EM_ECOG2, "Cyan Technology eCOG2 microprocessor" },
{ EM_SCORE, "Sunplus Score" },
{ EM_SCORE7, "Sunplus S+core7 RISC processor" },
{ EM_DSP24, "New Japan Radio (NJR) 24-bit DSP Processor" },
{ EM_VIDEOCORE3, "Broadcom VideoCore III processor" },
{ EM_LATTICEMICO32, "RISC processor for Lattice FPGA architecture" },
{ EM_SE_C17, "Seiko Epson C17 family" },
{ EM_TI_C6000, "Texas Instruments TMS320C6000 DSP family" },
{ EM_TI_C2000, "Texas Instruments TMS320C2000 DSP family" },
{ EM_TI_C5500, "Texas Instruments TMS320C55x DSP family" },
{ EM_res143, "Reserved" },
{ EM_res144, "Reserved" },
{ EM_res145, "Reserved" },
{ EM_res146, "Reserved" },
{ EM_res147, "Reserved" },
{ EM_res148, "Reserved" },
{ EM_res149, "Reserved" },
{ EM_res150, "Reserved" },
{ EM_res151, "Reserved" },
{ EM_res152, "Reserved" },
{ EM_res153, "Reserved" },
{ EM_res154, "Reserved" },
{ EM_res155, "Reserved" },
{ EM_res156, "Reserved" },
{ EM_res157, "Reserved" },
{ EM_res158, "Reserved" },
{ EM_res159, "Reserved" },
{ EM_MMDSP_PLUS, "STMicroelectronics 64bit VLIW Data Signal Processor" },
{ EM_CYPRESS_M8C, "Cypress M8C microprocessor" },
{ EM_R32C, "Renesas R32C series microprocessors" },
{ EM_TRIMEDIA, "NXP Semiconductors TriMedia architecture family" },
{ EM_QDSP6, "QUALCOMM DSP6 Processor" },
{ EM_8051, "Intel 8051 and variants" },
{ EM_STXP7X, "STMicroelectronics STxP7x family" },
{ EM_NDS32,
"Andes Technology compact code size embedded RISC processor family" },
{ EM_ECOG1, "Cyan Technology eCOG1X family" },
{ EM_ECOG1X, "Cyan Technology eCOG1X family" },
{ EM_MAXQ30, "Dallas Semiconductor MAXQ30 Core Micro-controllers" },
{ EM_XIMO16, "New Japan Radio (NJR) 16-bit DSP Processor" },
{ EM_MANIK, "M2000 Reconfigurable RISC Microprocessor" },
{ EM_CRAYNV2, "Cray Inc. NV2 vector architecture" },
{ EM_RX, "Renesas RX family" },
{ EM_METAG, "Imagination Technologies META processor architecture" },
{ EM_MCST_ELBRUS, "MCST Elbrus general purpose hardware architecture" },
{ EM_ECOG16, "Cyan Technology eCOG16 family" },
{ EM_CR16, "National Semiconductor CompactRISC 16-bit processor" },
{ EM_ETPU, "Freescale Extended Time Processing Unit" },
{ EM_SLE9X, "Infineon Technologies SLE9X core" },
{ EM_L1OM, "Intel L1OM" },
{ EM_INTEL181, "Reserved by Intel" },
{ EM_INTEL182, "Reserved by Intel" },
{ EM_AARCH64, "ARM AArch64" },
{ EM_res184, "Reserved by ARM" },
{ EM_AVR32, "Atmel Corporation 32-bit microprocessor family" },
{ EM_STM8, "STMicroeletronics STM8 8-bit microcontroller" },
{ EM_TILE64, "Tilera TILE64 multicore architecture family" },
{ EM_TILEPRO, "Tilera TILEPro multicore architecture family" },
{ EM_MICROBLAZE, "Xilinx MicroBlaze 32-bit RISC soft processor core" },
{ EM_CUDA, "NVIDIA CUDA architecture " },
{ EM_TILEGX, "Tilera TILE-Gx multicore architecture family" },
{ EM_CLOUDSHIELD, "CloudShield architecture family" },
{ EM_COREA_1ST, "KIPO-KAIST Core-A 1st generation processor family" },
{ EM_COREA_2ND, "KIPO-KAIST Core-A 2nd generation processor family" },
{ EM_ARC_COMPACT2, "Synopsys ARCompact V2" },
{ EM_OPEN8, "Open8 8-bit RISC soft processor core" },
{ EM_RL78, "Renesas RL78 family" },
{ EM_VIDEOCORE5, "Broadcom VideoCore V processor" },
{ EM_78KOR, "Renesas 78KOR family" },
{ EM_56800EX, "Freescale 56800EX Digital Signal Controller (DSC)" },
{ EM_BA1, "Beyond BA1 CPU architecture" },
{ EM_BA2, "Beyond BA2 CPU architecture" },
{ EM_XCORE, "XMOS xCORE processor family" },
{ EM_MCHP_PIC, "Microchip 8-bit PIC(r) family" },
{ EM_INTEL205, "Reserved by Intel" },
{ EM_INTEL206, "Reserved by Intel" },
{ EM_INTEL207, "Reserved by Intel" },
{ EM_INTEL208, "Reserved by Intel" },
{ EM_INTEL209, "Reserved by Intel" },
{ EM_KM32, "KM211 KM32 32-bit processor" },
{ EM_KMX32, "KM211 KMX32 32-bit processor" },
{ EM_KMX16, "KM211 KMX16 16-bit processor" },
{ EM_KMX8, "KM211 KMX8 8-bit processor" },
{ EM_KVARC, "KM211 KVARC processor" },
{ EM_CDP, "Paneve CDP architecture family" },
{ EM_COGE, "Cognitive Smart Memory Processor" },
{ EM_COOL, "iCelero CoolEngine" },
{ EM_NORC, "Nanoradio Optimized RISC" },
{ EM_CSR_KALIMBA, "CSR Kalimba architecture family" },
{ EM_Z80, "Zilog Z80" },
{ EM_VISIUM, "Controls and Data Services VISIUMcore processor" },
{ EM_FT32, "FTDI Chip FT32 high performance 32-bit RISC architecture" },
{ EM_MOXIE, "Moxie processor family" },
{ EM_AMDGPU, "AMD GPU architecture" },
{ EM_RISCV, "RISC-V" },
{ EM_LANAI, "Lanai processor" },
{ EM_CEVA, "CEVA Processor Architecture Family" },
{ EM_CEVA_X2, "CEVA X2 Processor Family" },
{ EM_BPF, "Linux BPF in-kernel virtual machine" },
{ EM_GRAPHCORE_IPU, "Graphcore Intelligent Processing Unit" },
{ EM_IMG1, "Imagination Technologies" },
{ EM_NFP, "Netronome Flow Processor (P)" },
{ EM_CSKY, "C-SKY processor family" },
{ EM_ARC_COMPACT3_64, "Synopsys ARCv2.3 64-bit" },
{ EM_MCS6502, "MOS Technology MCS 6502 processor" },
{ EM_ARC_COMPACT3, "Synopsys ARCv2.3 32-bit" },
{ EM_KVX, "Kalray VLIW core of the MPPA processor family" },
{ EM_65816, "WDC 65816/65C816" },
{ EM_LOONGARCH, "Loongson Loongarch" },
{ EM_KF32, "ChipON KungFu32" },
{ EM_MT, "Morpho Techologies MT processor" },
{ EM_ALPHA, "Alpha" },
{ EM_WEBASSEMBLY, "Web Assembly" },
{ EM_DLX, "OpenDLX" },
{ EM_XSTORMY16, "Sanyo XStormy16 CPU core" },
{ EM_IQ2000, "Vitesse IQ2000" },
{ EM_M32C_OLD, "M32C_OLD" },
{ EM_NIOS32, "Altera Nios" },
{ EM_CYGNUS_MEP, "Toshiba MeP Media Engine" },
{ EM_ADAPTEVA_EPIPHANY, "Adapteva EPIPHANY" },
{ EM_CYGNUS_FRV, "Fujitsu FR-V" },
{ EM_S12Z, "Freescale S12Z" },
};
static const struct section_type_table_t
{
const Elf64_Word key;
const char* str;
} section_type_table[] = {
{ SHT_NULL, "NULL" },
{ SHT_PROGBITS, "PROGBITS" },
{ SHT_SYMTAB, "SYMTAB" },
{ SHT_STRTAB, "STRTAB" },
{ SHT_RELA, "RELA" },
{ SHT_HASH, "HASH" },
{ SHT_DYNAMIC, "DYNAMIC" },
{ SHT_NOTE, "NOTE" },
{ SHT_NOBITS, "NOBITS" },
{ SHT_REL, "REL" },
{ SHT_SHLIB, "SHLIB" },
{ SHT_DYNSYM, "DYNSYM" },
{ SHT_INIT_ARRAY, "INIT_ARRAY" },
{ SHT_FINI_ARRAY, "FINI_ARRAY" },
{ SHT_PREINIT_ARRAY, "PREINIT_ARRAY" },
{ SHT_GROUP, "GROUP" },
{ SHT_SYMTAB_SHNDX, "SYMTAB_SHNDX" },
{ SHT_GNU_ATTRIBUTES, "GNU_ATTRIBUTES" },
{ SHT_GNU_HASH, "GNU_HASH" },
{ SHT_GNU_LIBLIST, "GNU_LIBLIST" },
{ SHT_CHECKSUM, "CHECKSUM" },
{ SHT_LOSUNW, "LOSUNW" },
{ SHT_SUNW_move, "SUNW_move" },
{ SHT_SUNW_COMDAT, "SUNW_COMDAT" },
{ SHT_SUNW_syminfo, "SUNW_syminfo" },
{ SHT_GNU_verdef, "GNU_verdef" },
{ SHT_GNU_verneed, "GNU_verneed" },
{ SHT_GNU_versym, "GNU_versym" },
{ SHT_ARM_EXIDX, "ARM_EXIDX" },
{ SHT_ARM_PREEMPTMAP, "ARM_PREEMPTMAP" },
{ SHT_ARM_ATTRIBUTES, "ARM_ATTRIBUTES" },
{ SHT_ARM_DEBUGOVERLAY, "ARM_DEBUGOVERLAY" },
{ SHT_ARM_OVERLAYSECTION, "ARM_OVERLAYSECTION" },
};
static const struct segment_type_table_t
{
const Elf_Word key;
const char* str;
} segment_type_table[] = {
{ PT_NULL, "NULL" },
{ PT_LOAD, "LOAD" },
{ PT_DYNAMIC, "DYNAMIC" },
{ PT_INTERP, "INTERP" },
{ PT_NOTE, "NOTE" },
{ PT_SHLIB, "SHLIB" },
{ PT_PHDR, "PHDR" },
{ PT_TLS, "TLS" },
{ PT_GNU_EH_FRAME, "GNU_EH_FRAME" },
{ PT_GNU_STACK, "GNU_STACK" },
{ PT_GNU_RELRO, "GNU_RELRO" },
{ PT_GNU_PROPERTY, "GNU_PROPERTY" },
{ PT_GNU_MBIND_LO, "GNU_MBIND_LO" },
{ PT_GNU_MBIND_HI, "GNU_MBIND_HI" },
{ PT_PAX_FLAGS, "PAX_FLAGS" },
{ PT_OPENBSD_RANDOMIZE, "OPENBSD_RANDOMIZE" },
{ PT_OPENBSD_WXNEEDED, "OPENBSD_WXNEEDED " },
{ PT_OPENBSD_BOOTDATA, "OPENBSD_BOOTDATA " },
{ PT_SUNWBSS, "PT_SUNWBSS" },
{ PT_SUNWSTACK, "SUNWSTACK" },
};
static const struct segment_flag_table_t
{
const Elf_Word key;
const char* str;
} segment_flag_table[] = {
{ 0, " " }, { 1, " E" }, { 2, " W " }, { 3, " WE" },
{ 4, "R " }, { 5, "R E" }, { 6, "RW " }, { 7, "RWE" },
};
static const struct symbol_bind_t
{
const Elf_Word key;
const char* str;
} symbol_bind_table[] = {
{ STB_LOCAL, "LOCAL" }, { STB_GLOBAL, "GLOBAL" },
{ STB_WEAK, "WEAK" }, { STB_LOOS, "LOOS" },
{ STB_HIOS, "HIOS" }, { STB_MULTIDEF, "MULTIDEF" },
{ STB_LOPROC, "LOPROC" }, { STB_HIPROC, "HIPROC" },
};
static const struct symbol_type_t
{
const Elf_Word key;
const char* str;
} symbol_type_table[] = {
{ STT_NOTYPE, "NOTYPE" }, { STT_OBJECT, "OBJECT" },
{ STT_FUNC, "FUNC" }, { STT_SECTION, "SECTION" },
{ STT_FILE, "FILE" }, { STT_COMMON, "COMMON" },
{ STT_TLS, "TLS" }, { STT_LOOS, "LOOS" },
{ STT_HIOS, "HIOS" }, { STT_LOPROC, "LOPROC" },
{ STT_HIPROC, "HIPROC" },
};
static const struct dynamic_tag_t
{
const Elf_Word key;
const char* str;
} dynamic_tag_table[] = {
{ DT_NULL, "NULL" },
{ DT_NEEDED, "NEEDED" },
{ DT_PLTRELSZ, "PLTRELSZ" },
{ DT_PLTGOT, "PLTGOT" },
{ DT_HASH, "HASH" },
{ DT_STRTAB, "STRTAB" },
{ DT_SYMTAB, "SYMTAB" },
{ DT_RELA, "RELA" },
{ DT_RELASZ, "RELASZ" },
{ DT_RELAENT, "RELAENT" },
{ DT_STRSZ, "STRSZ" },
{ DT_SYMENT, "SYMENT" },
{ DT_INIT, "INIT" },
{ DT_FINI, "FINI" },
{ DT_SONAME, "SONAME" },
{ DT_RPATH, "RPATH" },
{ DT_SYMBOLIC, "SYMBOLIC" },
{ DT_REL, "REL" },
{ DT_RELSZ, "RELSZ" },
{ DT_RELENT, "RELENT" },
{ DT_PLTREL, "PLTREL" },
{ DT_DEBUG, "DEBUG" },
{ DT_TEXTREL, "TEXTREL" },
{ DT_JMPREL, "JMPREL" },
{ DT_BIND_NOW, "BIND_NOW" },
{ DT_INIT_ARRAY, "INIT_ARRAY" },
{ DT_FINI_ARRAY, "FINI_ARRAY" },
{ DT_INIT_ARRAYSZ, "INIT_ARRAYSZ" },
{ DT_FINI_ARRAYSZ, "FINI_ARRAYSZ" },
{ DT_RUNPATH, "RUNPATH" },
{ DT_FLAGS, "FLAGS" },
{ DT_ENCODING, "ENCODING" },
{ DT_PREINIT_ARRAY, "PREINIT_ARRAY" },
{ DT_PREINIT_ARRAYSZ, "PREINIT_ARRAYSZ" },
{ DT_MAXPOSTAGS, "MAXPOSTAGS" },
{ DT_GNU_HASH, "GNU_HASH" },
{ DT_TLSDESC_PLT, "TLSDESC_PLT" },
{ DT_TLSDESC_GOT, "TLSDESC_GOT" },
{ DT_GNU_CONFLICT, "GNU_CONFLICT" },
{ DT_GNU_LIBLIST, "GNU_LIBLIST" },
{ DT_CONFIG, "CONFIG" },
{ DT_DEPAUDIT, "DEPAUDIT" },
{ DT_AUDIT, "AUDIT" },
{ DT_PLTPAD, "PLTPAD" },
{ DT_MOVETAB, "MOVETAB" },
{ DT_SYMINFO, "SYMINFO" },
{ DT_ADDRRNGHI, "ADDRRNGHI" },
{ DT_VERSYM, "VERSYM" },
{ DT_RELACOUNT, "RELACOUNT" },
{ DT_RELCOUNT, "RELCOUNT" },
{ DT_FLAGS_1, "FLAGS_1" },
{ DT_VERDEF, "VERDEF" },
{ DT_VERDEFNUM, "VERDEFNUM" },
{ DT_VERNEED, "VERNEED" },
{ DT_VERNEEDNUM, "VERNEEDNUM" },
};
// clang-format off
static const struct note_tag_t
{
struct note_values_t
{
Elf64_Word type;
std::string type_str;
std::string description;
};
std::string name;
std::vector<note_values_t> values;
} note_tag_table[] = {
{ "",
{ { NT_PRSTATUS, "NT_PRSTATUS", "prstatus struct" },
{ NT_FPREGSET, "NT_FPREGSET", "fpregset struct" },
{ NT_PRPSINFO, "NT_PRPSINFO", "prpsinfo struct" },
{ NT_TASKSTRUCT, "NT_TASKSTRUCT", "task struct" },
{ NT_AUXV, "NT_AUXV", "Elfxx_auxv_t" },
{ NT_PSTATUS, "NT_PSTATUS", "pstatus struct" },
{ NT_FPREGS, "NT_FPREGS", "fpregset struct" },
{ NT_PSINFO, "NT_PSINFO", "psinfo struct" },
{ NT_LWPSTATUS, "NT_LWPSTATUS", "lwpstatus_t struct" },
{ NT_LWPSINFO, "NT_LWPSINFO", "lwpsinfo_t struct" },
{ NT_WIN32PSTATUS, "NT_WIN32PSTATUS", "win32_pstatus struct" },
} },
{ "LINUX",
{ { NT_PRXFPREG, "NT_PRXFPREG", "Contains a user_xfpregs_struct;" },
{ NT_PPC_VMX, "NT_PPC_VMX", "PowerPC Altivec/VMX registers" },
{ NT_PPC_VSX, "NT_PPC_VSX", "PowerPC VSX registers" },
{ NT_PPC_TAR, "NT_PPC_TAR", "PowerPC Target Address Register" },
{ NT_PPC_PPR, "NT_PPC_PPR", "PowerPC Program Priority Register" },
{ NT_PPC_DSCR, "NT_PPC_DSCR", "PowerPC Data Stream Control Register" },
{ NT_PPC_EBB, "NT_PPC_EBB", "PowerPC Event Based Branch Registers" },
{ NT_PPC_PMU, "NT_PPC_PMU", "PowerPC Performance Monitor Registers" },
{ NT_PPC_TM_CGPR, "NT_PPC_TM_CGPR", "PowerPC TM checkpointed GPR Registers" },
{ NT_PPC_TM_CFPR, "NT_PPC_TM_CFPR", "PowerPC TM checkpointed FPR Registers" },
{ NT_PPC_TM_CVMX, "NT_PPC_TM_CVMX", "PowerPC TM checkpointed VMX Registers" },
{ NT_PPC_TM_CVSX, "NT_PPC_TM_CVSX", "PowerPC TM checkpointed VSX Registers" },
{ NT_PPC_TM_SPR, "NT_PPC_TM_SPR", "PowerPC TM Special Purpose Registers" },
{ NT_PPC_TM_CTAR, "NT_PPC_TM_CTAR", "PowerPC TM checkpointed TAR" },
{ NT_PPC_TM_CPPR, "NT_PPC_TM_CPPR", "PowerPC TM checkpointed PPR" },
{ NT_PPC_TM_CDSCR, "NT_PPC_TM_CDSCR", "PowerPC TM checkpointed Data SCR" },
{ NT_386_TLS, "NT_386_TLS", "x86 TLS information" },
{ NT_386_IOPERM, "NT_386_IOPERM", "x86 io permissions" },
{ NT_X86_XSTATE, "NT_X86_XSTATE", "x86 XSAVE extended state" },
{ NT_X86_CET, "NT_X86_CET", "x86 CET state" },
{ NT_S390_HIGH_GPRS, "NT_S390_HIGH_GPRS", "S/390 upper halves of GPRs " },
{ NT_S390_TIMER, "NT_S390_TIMER", "S390 timer" },
{ NT_S390_TODCMP, "NT_S390_TODCMP", "S390 TOD clock comparator" },
{ NT_S390_TODPREG, "NT_S390_TODPREG", "S390 TOD programmable register" },
{ NT_S390_CTRS, "NT_S390_CTRS", "S390 control registers" },
{ NT_S390_PREFIX, "NT_S390_PREFIX", "S390 prefix register" },
{ NT_S390_LAST_BREAK, "NT_S390_LAST_BREAK", "S390 breaking event address" },
{ NT_S390_SYSTEM_CALL, "NT_S390_SYSTEM_CALL", "S390 system call restart data" },
{ NT_S390_TDB, "NT_S390_TDB", "S390 transaction diagnostic block" },
{ NT_S390_VXRS_LOW, "NT_S390_VXRS_LOW", "S390 vector registers 0-15 upper half" },
{ NT_S390_VXRS_HIGH, "NT_S390_VXRS_HIGH", "S390 vector registers 16-31" },
{ NT_S390_GS_CB, "NT_S390_GS_CB", "s390 guarded storage registers" },
{ NT_S390_GS_BC, "NT_S390_GS_BC", "s390 guarded storage broadcast control block" },
{ NT_ARM_VFP, "NT_ARM_VFP", "ARM VFP registers" },
{ NT_ARM_TLS, "NT_ARM_TLS", "AArch TLS registers" },
{ NT_ARM_HW_BREAK, "NT_ARM_HW_BREAK", "AArch hardware breakpoint registers" },
{ NT_ARM_HW_WATCH, "NT_ARM_HW_WATCH", "AArch hardware watchpoint registers" },
{ NT_ARM_SVE, "NT_ARM_SVE", "AArch SVE registers. " },
{ NT_ARM_PAC_MASK, "NT_ARM_PAC_MASK", "AArch pointer authentication code masks" },
{ NT_ARM_PACA_KEYS, "NT_ARM_PACA_KEYS", "ARM pointer authentication address keys" },
{ NT_ARM_PACG_KEYS, "NT_ARM_PACG_KEYS", "ARM pointer authentication generic keys" },
{ NT_ARM_TAGGED_ADDR_CTRL, "NT_ARM_TAGGED_ADDR_CTRL", "AArch64 tagged address control (prctl())" },
{ NT_ARM_PAC_ENABLED_KEYS, "NT_ARM_PAC_ENABLED_KEYS", "AArch64 pointer authentication enabled keys (prctl())" },
{ NT_ARC_V2, "NT_ARC_V2", "ARC HS accumulator/extra registers. " },
{ NT_LARCH_CPUCFG, "NT_LARCH_CPUCFG", "LoongArch CPU config registers" },
{ NT_LARCH_CSR, "NT_LARCH_CSR", "LoongArch Control State Registers" },
{ NT_LARCH_LSX, "NT_LARCH_LSX", "LoongArch SIMD eXtension registers" },
{ NT_LARCH_LASX, "NT_LARCH_LASX", "LoongArch Advanced SIMD eXtension registers" },
{ NT_RISCV_CSR, "NT_RISCV_CSR", "RISC-V Control and Status Registers" },
} },
{ "CORE",
{ { NT_LARCH_LBT, "NT_LARCH_LBT", "LoongArch Binary Translation registers" }
} },
{ "FreeBSD",
{ { NT_FREEBSD_THRMISC, "NT_FREEBSD_THRMISC", "Thread miscellaneous info." },
{ NT_FREEBSD_PROCSTAT_PROC, "NT_FREEBSD_PROCSTAT_PROC", "Procstat proc data." },
{ NT_FREEBSD_PROCSTAT_FILES, "NT_FREEBSD_PROCSTAT_FILES", "Procstat files data." },
{ NT_FREEBSD_PROCSTAT_VMMAP, "NT_FREEBSD_PROCSTAT_VMMAP", "Procstat vmmap data." },
{ NT_FREEBSD_PROCSTAT_GROUPS, "NT_FREEBSD_PROCSTAT_GROUPS", "Procstat groups data." },
{ NT_FREEBSD_PROCSTAT_UMASK, "NT_FREEBSD_PROCSTAT_UMASK", "Procstat umask data." },
{ NT_FREEBSD_PROCSTAT_RLIMIT, "NT_FREEBSD_PROCSTAT_RLIMIT", "Procstat rlimit data." },
{ NT_FREEBSD_PROCSTAT_OSREL, "NT_FREEBSD_PROCSTAT_OSREL", "Procstat osreldate data." },
{ NT_FREEBSD_PROCSTAT_PSSTRINGS, "NT_FREEBSD_PROCSTAT_PSSTRINGS", "Procstat ps_strings data." },
{ NT_FREEBSD_PROCSTAT_AUXV, "NT_FREEBSD_PROCSTAT_AUXV", "Procstat auxv data." },
{ NT_FREEBSD_PTLWPINFO, "NT_FREEBSD_PTLWPINFO", "Thread ptrace miscellaneous info." },
} },
{ "NetBSD-CORE",
{ { NT_NETBSDCORE_PROCINFO, "NT_NETBSDCORE_PROCINFO", "Has a struct procinfo" },
{ NT_NETBSDCORE_AUXV, "NT_NETBSDCORE_AUXV", "Has auxv data" },
{ NT_NETBSDCORE_LWPSTATUS, "NT_NETBSDCORE_LWPSTATUS", "Has LWPSTATUS data" },
{ NT_NETBSDCORE_FIRSTMACH, "NT_NETBSDCORE_FIRSTMACH", "start of machdep note types" },
} },
{ "OpenBSD",
{ { NT_OPENBSD_PROCINFO, "NT_OPENBSD_PROCINFO", "" },
{ NT_OPENBSD_AUXV, "NT_OPENBSD_AUXV", "" },
{ NT_OPENBSD_REGS, "NT_OPENBSD_REGS", "" },
{ NT_OPENBSD_FPREGS, "NT_OPENBSD_FPREGS", "" },
{ NT_OPENBSD_XFPREGS, "NT_OPENBSD_XFPREGS", "" },
{ NT_OPENBSD_WCOOKIE, "NT_OPENBSD_WCOOKIE", "" },
} },
{ "SPU",
{ { NT_SPU, "NT_SPU", "" }
} },
{ "GNU",
{
{ NT_GNU_ABI_TAG, "NT_GNU_ABI_TAG", "GNU ABI tag" },
{ NT_GNU_HWCAP, "NT_GNU_HWCAP", "Used by ld.so and kernel vDSO" },
{ NT_GNU_BUILD_ID, "NT_GNU_BUILD_ID", "Build ID of the binary" },
{ NT_GNU_GOLD_VERSION, "NT_GNU_GOLD_VERSION", "Version of GNU gold used to link the binary" },
{ NT_GNU_PROPERTY_TYPE_0, "NT_GNU_PROPERTY_TYPE_0", "Property type 0" },
// { NT_GNU_PROPERTY_TYPE_1, "NT_GNU_PROPERTY_TYPE_1", "Property type 1" },
// { NT_GNU_PROPERTY_TYPE_2, "NT_GNU_PROPERTY_TYPE_2", "Property type 2" },
// { NT_GNU_PROPERTY_TYPE_3, "NT_GNU_PROPERTY_TYPE_3", "Property type 3" },
// { NT_GNU_PROPERTY_TYPE_4, "NT_GNU_PROPERTY_TYPE_4", "Property type 4" },
// { NT_GNU_PROPERTY_TYPE_5, "NT_GNU_PROPERTY_TYPE_5", "Property type 5" },
// { NT_GNU_PROPERTY_TYPE_6, "NT_GNU_PROPERTY_TYPE_6", "Property type 6" },
// { NT_GNU_PROPERTY_TYPE_7, "NT_GNU_PROPERTY_TYPE_7", "Property type 7" },
// { NT_GNU_PROPERTY_TYPE_8, "NT_GNU_PROPERTY_TYPE_8", "Property type 8" },
// { NT_GNU_PROPERTY_TYPE_9, "NT_GNU_PROPERTY_TYPE_9", "Property type 9" },
// { NT_GNU_PROPERTY_TYPE_10, "NT_GNU_PROPERTY_TYPE_10", "Property type 10" },
// { NT_GNU_PROPERTY_TYPE_11, "NT_GNU_PROPERTY_TYPE_11", "Property type 11" },
// { NT_GNU_PROPERTY_TYPE_12, "NT_GNU_PROPERTY_TYPE_12", "Property type 12" },
// { NT_GNU_PROPERTY_TYPE_13, "NT_GNU_PROPERTY_TYPE_13", "Property type 13" },
// { NT_GNU_PROPERTY_TYPE_14, "NT_GNU_PROPERTY_TYPE_14", "Property type 14" },
} },
// { "SOLARIS",
// { { NT_SOLARIS_AUXV, "NT_SOLARIS_AUXV", "" }
// } },
// { "AIX",
// { { NT_AIX_AUXV, "NT_AIX_AUXV", "" }
// } },
// { "IRIX",
// { { NT_IRIX_FPREGS, "NT_IRIX_FPREGS", "" }
// } },
};
// clang-format on
static const ELFIO::Elf_Xword MAX_DATA_ENTRIES = 64;
//------------------------------------------------------------------------------
class dump
{
#define DUMP_DEC_FORMAT( width ) \
std::setw( width ) << std::setfill( ' ' ) << std::dec << std::right
#define DUMP_HEX0x_FORMAT( width ) \
"0x" << std::setw( width ) << std::setfill( '0' ) << std::hex << std::right
#define DUMP_HEX_FORMAT( width ) \
std::setw( width ) << std::setfill( '0' ) << std::hex << std::right
#define DUMP_STR_FORMAT( width ) \
std::setw( width ) << std::setfill( ' ' ) << std::hex << std::left
public:
//------------------------------------------------------------------------------
static void header( std::ostream& out, const elfio& reader )
{
if ( !reader.get_header_size() ) {
return;
}
out << "ELF Header" << std::endl
<< std::endl
<< " Class: " << str_class( reader.get_class() ) << std::endl
<< " Encoding: " << str_endian( reader.get_encoding() )
<< std::endl
<< " ELFVersion: " << str_version( reader.get_elf_version() )
<< std::endl
<< " OS/ABI: " << str_os_abi( reader.get_os_abi() )
<< std::endl
<< " ABI Version:" << (int)reader.get_abi_version() << std::endl
<< " Type: " << str_type( reader.get_type() ) << std::endl
<< " Machine: " << str_machine( reader.get_machine() )
<< std::endl
<< " Version: " << str_version( reader.get_version() )
<< std::endl
<< " Entry: "
<< "0x" << std::hex << reader.get_entry() << std::endl
<< " Flags: "
<< "0x" << std::hex << reader.get_flags() << std::endl
<< std::endl;
}
//------------------------------------------------------------------------------
static void section_headers( std::ostream& out, const elfio& reader )
{
Elf_Half n = reader.sections.size();
if ( n == 0 ) {
return;
}
out << "Section Headers:" << std::endl;
if ( reader.get_class() == ELFCLASS32 ) { // Output for 32-bit
out << "[ Nr ] Type Addr Size ES Flg "
"Lk Inf Al Name"
<< std::endl;
}
else { // Output for 64-bit
out << "[ Nr ] Type Addr Size "
" Offset Flg"
<< std::endl
<< " ES Lk Inf Al Name" << std::endl;
}
for ( Elf_Half i = 0; i < n; ++i ) { // For all sections
const section* sec = reader.sections[i];
section_header( out, i, sec, reader.get_class() );
}
out << "Key to Flags: W (write), A (alloc), X (execute), " << std::endl;
out << " M (merge), S (strings), I (info)," << std::endl;
out << " L (link order), O (extra OS processing required),"
<< std::endl;
out << " G (group), T (TLS), C (compressed), E (exclude)"
<< std::endl;
}
//------------------------------------------------------------------------------
static void section_header( std::ostream& out,
Elf_Half no,
const section* sec,
unsigned char elf_class )
{
std::ios_base::fmtflags original_flags = out.flags();
// clang-format off
if ( elf_class == ELFCLASS32 ) { // Output for 32-bit
out << "[" << DUMP_DEC_FORMAT( 5 ) << no << "] "
<< DUMP_STR_FORMAT( 17 ) << str_section_type( sec->get_type() )
<< " " << DUMP_HEX0x_FORMAT( 8 ) << sec->get_address() << " "
<< DUMP_HEX0x_FORMAT( 8 ) << sec->get_size() << " "
<< DUMP_HEX0x_FORMAT( 2 ) << sec->get_entry_size() << " "
<< DUMP_STR_FORMAT( 3 ) << section_flags( sec->get_flags() )
<< " " << DUMP_HEX0x_FORMAT( 2 ) << sec->get_link() << " "
<< DUMP_HEX0x_FORMAT( 3 ) << sec->get_info() << " "
<< DUMP_HEX0x_FORMAT( 2 ) << sec->get_addr_align() << " "
<< DUMP_STR_FORMAT( 17 ) << sec->get_name() << " " << std::endl;
}
else { // Output for 64-bit
out << "[" << DUMP_DEC_FORMAT( 5 ) << no << "] "
<< DUMP_STR_FORMAT( 17 ) << str_section_type( sec->get_type() ) << " "
<< DUMP_HEX0x_FORMAT( 16 ) << sec->get_address() << " "
<< DUMP_HEX0x_FORMAT( 16 ) << sec->get_size() << " "
<< DUMP_HEX0x_FORMAT( 8 ) << sec->get_offset() << " "
<< DUMP_STR_FORMAT( 3) << section_flags( sec->get_flags() )
<< std::endl
<< DUMP_STR_FORMAT( 8 ) << " "
<< DUMP_HEX0x_FORMAT( 4 ) << sec->get_entry_size() << " "
<< DUMP_HEX0x_FORMAT( 4 ) << sec->get_link() << " "
<< DUMP_HEX0x_FORMAT( 4 ) << sec->get_info() << " "
<< DUMP_HEX0x_FORMAT( 4 ) << sec->get_addr_align() << " "
<< DUMP_STR_FORMAT( 17 ) << sec->get_name()
<< std::endl;
}
// clang-format on
out.flags( original_flags );
return;
}
//------------------------------------------------------------------------------
static void segment_headers( std::ostream& out, const elfio& reader )
{
Elf_Half n = reader.segments.size();
if ( n == 0 ) {
return;
}
out << "Program Headers:" << std::endl;
if ( reader.get_class() == ELFCLASS32 ) { // Output for 32-bit
out << "[ Nr ] Type VirtAddr PhysAddr FileSize "
"Mem.Size Flags Align"
<< std::endl;
}
else { // Output for 64-bit
out << "[ Nr ] Type Offset VirtAddr "
" PhysAddr"
<< std::endl
<< " FileSize MemSize "
" Flags Align"
<< std::endl;
}
for ( Elf_Half i = 0; i < n; ++i ) {
const segment* seg = reader.segments[i];
segment_header( out, i, seg, reader.get_class() );
}
out << std::endl;
}
//------------------------------------------------------------------------------
static void segment_header( std::ostream& out,
Elf_Half no,
const segment* seg,
unsigned int elf_class )
{
std::ios_base::fmtflags original_flags = out.flags();
// clang-format off
if ( elf_class == ELFCLASS32 ) { // Output for 32-bit
out << "[" << DUMP_DEC_FORMAT( 5 ) << no << "] "
<< DUMP_STR_FORMAT( 14 ) << str_segment_type( seg->get_type() )
<< " " << DUMP_HEX0x_FORMAT( 8 ) << seg->get_virtual_address()
<< " " << DUMP_HEX0x_FORMAT( 8 ) << seg->get_physical_address()
<< " " << DUMP_HEX0x_FORMAT( 8 ) << seg->get_file_size() << " "
<< DUMP_HEX0x_FORMAT( 8 ) << seg->get_memory_size() << " "
<< DUMP_STR_FORMAT( 8 ) << str_segment_flag( seg->get_flags() )
<< " " << DUMP_HEX0x_FORMAT( 8 ) << seg->get_align() << " "
<< std::endl;
}
else { // Output for 64-bit
out << "[" << DUMP_DEC_FORMAT( 5 ) << no << "] "
<< DUMP_STR_FORMAT( 14 ) << str_segment_type( seg->get_type() ) << " "
<< DUMP_HEX0x_FORMAT( 16 ) << seg->get_offset() << " "
<< DUMP_HEX0x_FORMAT( 16 ) << seg->get_virtual_address() << " "
<< DUMP_HEX0x_FORMAT( 16 ) << seg->get_physical_address()
<< std::endl
<< DUMP_STR_FORMAT( 23 ) << " "
<< DUMP_HEX0x_FORMAT( 16 ) << seg->get_file_size() << " "
<< DUMP_HEX0x_FORMAT( 16 ) << seg->get_memory_size() << " "
<< DUMP_STR_FORMAT( 3 ) << str_segment_flag( seg->get_flags() ) << " "
<< DUMP_HEX0x_FORMAT( 1 ) << seg->get_align()
<< std::endl;
}
// clang-format on
out.flags( original_flags );
}
//------------------------------------------------------------------------------
static void symbol_tables( std::ostream& out, const elfio& reader )
{
for ( const auto& sec : reader.sections ) { // For all sections
if ( SHT_SYMTAB == sec->get_type() ||
SHT_DYNSYM == sec->get_type() ) {
const_symbol_section_accessor symbols( reader, sec.get() );
Elf_Xword sym_no = symbols.get_symbols_num();
if ( sym_no == 0 ) {
continue;
}
out << "Symbol table (" << sec->get_name() << ")" << std::endl;
if ( reader.get_class() == ELFCLASS32 ) { // Output for 32-bit
out << "[ Nr ] Value Size Type Bind "
" Sect Name"
<< std::endl;
}
else { // Output for 64-bit
out << "[ Nr ] Value Size "
"Type Bind Sect"
<< std::endl
<< " Name" << std::endl;
}
for ( Elf_Xword i = 0; i < sym_no; ++i ) {
std::string name;
Elf64_Addr value = 0;
Elf_Xword size = 0;
unsigned char bind = 0;
unsigned char type = 0;
Elf_Half section = 0;
unsigned char other = 0;
symbols.get_symbol( i, name, value, size, bind, type,
section, other );
symbol_table( out, i, name, value, size, bind, type,
section, reader.get_class() );
}
out << std::endl;
}
}
}
//------------------------------------------------------------------------------
static void symbol_table( std::ostream& out,
Elf_Xword no,
const std::string& name,
Elf64_Addr value,
Elf_Xword size,
unsigned char bind,
unsigned char type,
Elf_Half section,
unsigned int elf_class )
{
std::ios_base::fmtflags original_flags = out.flags();
if ( elf_class == ELFCLASS32 ) { // Output for 32-bit
out << "[" << DUMP_DEC_FORMAT( 5 ) << no << "] "
<< DUMP_HEX0x_FORMAT( 8 ) << value << " "
<< DUMP_HEX0x_FORMAT( 8 ) << size << " " << DUMP_STR_FORMAT( 7 )
<< str_symbol_type( type ) << " " << DUMP_STR_FORMAT( 8 )
<< str_symbol_bind( bind ) << " " << DUMP_DEC_FORMAT( 5 )
<< section << " " << DUMP_STR_FORMAT( 1 ) << name << " "
<< std::endl;
}
else { // Output for 64-bit
out << "[" << DUMP_DEC_FORMAT( 5 ) << no << "] "
<< DUMP_HEX0x_FORMAT( 16 ) << value << " "
<< DUMP_HEX0x_FORMAT( 16 ) << size << " "
<< DUMP_STR_FORMAT( 7 ) << str_symbol_type( type ) << " "
<< DUMP_STR_FORMAT( 8 ) << str_symbol_bind( bind ) << " "
<< DUMP_DEC_FORMAT( 5 ) << section << " " << std::endl
<< " " << DUMP_STR_FORMAT( 1 ) << name << " "
<< std::endl;
}
out.flags( original_flags );
}
//------------------------------------------------------------------------------
static void notes( std::ostream& out, const elfio& reader )
{
for ( const auto& sec : reader.sections ) { // For all sections
if ( SHT_NOTE == sec->get_type() ) { // Look at notes
note_section_accessor notes( reader, sec.get() );
Elf_Word no_notes = notes.get_notes_num();
if ( no_notes == 0 )
continue;
out << "Note section (" << sec->get_name() << ")" << std::endl
<< " No Name Data size Description"
<< std::endl;
for ( Elf_Word j = 0; j < no_notes; ++j ) { // For all notes
Elf_Word type;
std::string name;
char* desc;
Elf_Word descsz;
if ( notes.get_note( j, type, name, desc, descsz ) ) {
// 'name' usually contains \0 at the end. Remove it
name = name.c_str();
note( out, j, type, name, desc, descsz );
out << std::endl;
}
}
out << std::endl;
}
}
Elf_Half no = reader.segments.size();
for ( Elf_Half i = 0; i < no; ++i ) { // For all segments
segment* seg = reader.segments[i];
if ( PT_NOTE == seg->get_type() ) { // Look at notes
note_segment_accessor notes( reader, seg );
Elf_Word no_notes = notes.get_notes_num();
if ( no_notes == 0 )
continue;
out << "Note segment (" << i << ")" << std::endl
<< " No Name Data size Description"
<< std::endl;
for ( Elf_Word j = 0; j < no_notes; ++j ) { // For all notes
Elf_Word type;
std::string name;
char* desc;
Elf_Word descsz;
if ( notes.get_note( j, type, name, desc, descsz ) ) {
// 'name' usually contains \0 at the end. Remove it
name = name.c_str();
note( out, j, type, name, desc, descsz );
out << std::endl;
}
}
out << std::endl;
}
}
}
//------------------------------------------------------------------------------
static void note( std::ostream& out,
int no,
Elf_Word type,
const std::string& name,
void* desc,
Elf_Word descsz )
{
out << " [" << DUMP_DEC_FORMAT( 2 ) << no << "] ";
const auto name_group = std::find_if(
std::begin( note_tag_table ), std::end( note_tag_table ),
[&name]( const note_tag_t& entry ) { return entry.name == name; } );
std::vector<note_tag_t::note_values_t>::const_iterator type_value;
if ( name_group != std::end( note_tag_table ) ) {
type_value = std::find_if(
name_group->values.begin(), name_group->values.end(),
[&type]( const note_tag_t::note_values_t& e ) {
return e.type == type;
} );
}
if ( name_group != std::end( note_tag_table ) &&
type_value != name_group->values.end() ) {
out << DUMP_STR_FORMAT( 12 ) << name_group->name << " "
<< DUMP_HEX0x_FORMAT( 8 ) << descsz << " "
<< type_value->type_str << " (" << type_value->description
<< ")";
}
else {
out << DUMP_STR_FORMAT( 12 ) << name << " "
<< DUMP_HEX0x_FORMAT( 8 ) << descsz << " "
<< DUMP_HEX0x_FORMAT( 8 ) << type;
}
if ( descsz != 0 ) {
for ( Elf_Word i = 0; i < descsz; ++i ) {
if ( i % 16 == 0 ) {
out << std::endl << " ";
}
out << DUMP_HEX_FORMAT( 2 )
<< (uint32_t)( (uint8_t*)( desc ) )[i];
}
}
}
//------------------------------------------------------------------------------
static void modinfo( std::ostream& out, const elfio& reader )
{
for ( const auto& sec : reader.sections ) { // For all sections
if ( ".modinfo" == sec->get_name() ) { // Look for the section
out << "Section .modinfo" << std::endl;
const_modinfo_section_accessor modinfo( sec.get() );
for ( Elf_Word i = 0; i < modinfo.get_attribute_num(); i++ ) {
std::string field;
std::string value;
if ( modinfo.get_attribute( i, field, value ) ) {
out << " " << std::setw( 20 ) << field
<< std::setw( 0 ) << " = " << value << std::endl;
}
}
out << std::endl;
break;
}
}
}
//------------------------------------------------------------------------------
static void dynamic_tags( std::ostream& out, const elfio& reader )
{
for ( const auto& sec : reader.sections ) { // For all sections
if ( SHT_DYNAMIC == sec->get_type() ) {
dynamic_section_accessor dynamic( reader, sec.get() );
Elf_Xword dyn_no = dynamic.get_entries_num();
if ( dyn_no == 0 )
continue;
out << "Dynamic section (" << sec->get_name() << ")"
<< std::endl;
out << "[ Nr ] Tag Name/Value" << std::endl;
for ( Elf_Xword i = 0; i < dyn_no; ++i ) {
Elf_Xword tag = 0;
Elf_Xword value = 0;
std::string str;
dynamic.get_entry( i, tag, value, str );
dynamic_tag( out, i, tag, value, str, reader.get_class() );
if ( DT_NULL == tag ) {
break;
}
}
out << std::endl;
}
}
}
//------------------------------------------------------------------------------
static void dynamic_tag( std::ostream& out,
Elf_Xword no,
Elf_Xword tag,
Elf_Xword value,
const std::string& str,
unsigned int /*elf_class*/ )
{
out << "[" << DUMP_DEC_FORMAT( 5 ) << no << "] "
<< DUMP_STR_FORMAT( 16 ) << str_dynamic_tag( tag ) << " ";
if ( str.empty() ) {
out << DUMP_HEX0x_FORMAT( 16 ) << value << " ";
}
else {
out << DUMP_STR_FORMAT( 32 ) << str << " ";
}
out << std::endl;
}
//------------------------------------------------------------------------------
static void section_data( std::ostream& out, const section* sec )
{
std::ios_base::fmtflags original_flags = out.flags();
out << sec->get_name() << std::endl;
const char* pdata = sec->get_data();
if ( pdata ) {
ELFIO::Elf_Xword i;
for ( i = 0; i < std::min( sec->get_size(), MAX_DATA_ENTRIES );
++i ) {
if ( i % 16 == 0 ) {
out << "[" << DUMP_HEX0x_FORMAT( 8 ) << i << "]";
}
out << " " << DUMP_HEX0x_FORMAT( 2 )
<< ( pdata[i] & 0x000000FF );
if ( i % 16 == 15 ) {
out << std::endl;
}
}
if ( i % 16 != 0 ) {
out << std::endl;
}
out.flags( original_flags );
}
return;
}
//------------------------------------------------------------------------------
static void section_datas( std::ostream& out, const elfio& reader )
{
Elf_Half n = reader.sections.size();
if ( n == 0 ) {
return;
}
out << "Section Data:" << std::endl;
for ( Elf_Half i = 1; i < n; ++i ) { // For all sections
const section* sec = reader.sections[i];
if ( sec->get_type() == SHT_NOBITS ) {
continue;
}
section_data( out, sec );
}
out << std::endl;
}
//------------------------------------------------------------------------------
static void
segment_data( std::ostream& out, Elf_Half no, const segment* seg )
{
std::ios_base::fmtflags original_flags = out.flags();
out << "Segment # " << no << std::endl;
const char* pdata = seg->get_data();
if ( pdata ) {
ELFIO::Elf_Xword i;
for ( i = 0; i < std::min( seg->get_file_size(), MAX_DATA_ENTRIES );
++i ) {
if ( i % 16 == 0 ) {
out << "[" << DUMP_HEX0x_FORMAT( 8 ) << i << "]";
}
out << " " << DUMP_HEX0x_FORMAT( 2 )
<< ( pdata[i] & 0x000000FF );
if ( i % 16 == 15 ) {
out << std::endl;
}
}
if ( i % 16 != 0 ) {
out << std::endl;
}
out.flags( original_flags );
}
return;
}
//------------------------------------------------------------------------------
static void segment_datas( std::ostream& out, const elfio& reader )
{
Elf_Half n = reader.segments.size();
if ( n == 0 ) {
return;
}
out << "Segment Data:" << std::endl;
for ( Elf_Half i = 0; i < n; ++i ) { // For all sections
const segment* seg = reader.segments[i];
segment_data( out, i, seg );
}
out << std::endl;
}
//------------------------------------------------------------------------------
#define STR_FUNC_TABLE( name ) \
template <typename T> static std::string str_##name( const T key ) \
{ \
return format_assoc( name##_table, key ); \
}
STR_FUNC_TABLE( class )
STR_FUNC_TABLE( endian )
STR_FUNC_TABLE( version )
STR_FUNC_TABLE( os_abi )
STR_FUNC_TABLE( type )
STR_FUNC_TABLE( machine )
STR_FUNC_TABLE( section_type )
STR_FUNC_TABLE( segment_type )
STR_FUNC_TABLE( segment_flag )
STR_FUNC_TABLE( symbol_bind )
STR_FUNC_TABLE( symbol_type )
STR_FUNC_TABLE( dynamic_tag )
#undef STR_FUNC_TABLE
private:
//------------------------------------------------------------------------------
template <typename T, typename K>
std::string static find_value_in_table( const T& table, const K& key )
{
std::string res = "?";
for ( unsigned int i = 0; i < sizeof( table ) / sizeof( table[0] );
++i ) {
if ( table[i].key == key ) {
res = table[i].str;
break;
}
}
return res;
}
//------------------------------------------------------------------------------
template <typename T, typename K>
static std::string format_assoc( const T& table, const K& key )
{
std::string str = find_value_in_table( table, key );
if ( str == "?" ) {
std::ostringstream oss;
oss << str << " (0x" << std::hex << key << ")";
str = oss.str();
}
return str;
}
//------------------------------------------------------------------------------
template <typename T>
static std::string format_assoc( const T& table, const char key )
{
return format_assoc( table, (const int)key );
}
//------------------------------------------------------------------------------
static std::string section_flags( Elf_Xword flags )
{
std::string ret = "";
if ( flags & SHF_WRITE ) {
ret += "W";
}
if ( flags & SHF_ALLOC ) {
ret += "A";
}
if ( flags & SHF_EXECINSTR ) {
ret += "X";
}
if ( flags & SHF_MERGE ) {
ret += "M";
}
if ( flags & SHF_STRINGS ) {
ret += "S";
}
if ( flags & SHF_INFO_LINK ) {
ret += "I";
}
if ( flags & SHF_LINK_ORDER ) {
ret += "L";
}
if ( flags & SHF_OS_NONCONFORMING ) {
ret += "O";
}
if ( flags & SHF_GROUP ) {
ret += "G";
}
if ( flags & SHF_TLS ) {
ret += "T";
}
if ( flags & SHF_COMPRESSED ) {
ret += "C";
}
if ( flags & SHF_EXCLUDE ) {
ret += "E";
}
if ( flags & SHF_GNU_MBIND ) {
ret += "D";
}
return ret;
}
#undef DUMP_DEC_FORMAT
#undef DUMP_HEX0x_FORMAT
#undef DUMP_STR_FORMAT
}; // class dump
} // namespace ELFIO
#endif // ELFIO_DUMP_HPP