1 /* 2 Copyright (C) 2001-present by Serge Lamikhov-Center 3 4 Permission is hereby granted, free of charge, to any person obtaining a copy 5 of this software and associated documentation files (the "Software"), to deal 6 in the Software without restriction, including without limitation the rights 7 to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 8 copies of the Software, and to permit persons to whom the Software is 9 furnished to do so, subject to the following conditions: 10 11 The above copyright notice and this permission notice shall be included in 12 all copies or substantial portions of the Software. 13 14 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 15 IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 16 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 17 AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 18 LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 19 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 20 THE SOFTWARE. 21 */ 22 23 #ifndef ELFIO_DUMP_HPP 24 #define ELFIO_DUMP_HPP 25 26 #include <algorithm> 27 #include <string> 28 #include <ostream> 29 #include <sstream> 30 #include <iomanip> 31 #include <elfio/elfio.hpp> 32 33 namespace ELFIO { 34 35 static const struct class_table_t 36 { 37 const char key; 38 const char* str; 39 } class_table[] = { 40 { ELFCLASS32, "ELF32" }, 41 { ELFCLASS64, "ELF64" }, 42 }; 43 44 static const struct endian_table_t 45 { 46 const char key; 47 const char* str; 48 } endian_table[] = { 49 { ELFDATANONE, "None" }, 50 { ELFDATA2LSB, "Little endian" }, 51 { ELFDATA2MSB, "Big endian" }, 52 }; 53 54 static const struct os_abi_table_t 55 { 56 const unsigned char key; 57 const char* str; 58 } os_abi_table[] = { 59 { ELFOSABI_NONE, "UNIX System V" }, 60 { ELFOSABI_HPUX, "Hewlett-Packard HP-UX" }, 61 { ELFOSABI_NETBSD, "NetBSD" }, 62 { ELFOSABI_LINUX, "Linux" }, 63 { ELFOSABI_HURD, "GNU Hurd" }, 64 { ELFOSABI_SOLARIS, "Sun Solaris" }, 65 { ELFOSABI_AIX, "AIX" }, 66 { ELFOSABI_IRIX, "IRIX" }, 67 { ELFOSABI_FREEBSD, "FreeBSD" }, 68 { ELFOSABI_TRU64, "Compaq TRU64 UNIX" }, 69 { ELFOSABI_MODESTO, "Novell Modesto" }, 70 { ELFOSABI_OPENBSD, "Open BSD" }, 71 { ELFOSABI_OPENVMS, "Open VMS" }, 72 { ELFOSABI_NSK, "Hewlett-Packard Non-Stop Kernel" }, 73 { ELFOSABI_AROS, "Amiga Research OS" }, 74 { ELFOSABI_FENIXOS, "FenixOS" }, 75 { ELFOSABI_NUXI, "Nuxi CloudABI" }, 76 { ELFOSABI_OPENVOS, "OpenVOS" }, 77 { ELFOSABI_ARM, "ARM" }, 78 { ELFOSABI_STANDALONE, "Standalone (embedded)" }, 79 }; 80 81 static const struct version_table_t 82 { 83 const Elf64_Word key; 84 const char* str; 85 } version_table[] = { 86 { EV_NONE, "None" }, 87 { EV_CURRENT, "Current" }, 88 }; 89 90 static const struct type_table_t 91 { 92 const Elf32_Half key; 93 const char* str; 94 } type_table[] = { 95 { ET_NONE, "No file type" }, { ET_REL, "Relocatable file" }, 96 { ET_EXEC, "Executable file" }, { ET_DYN, "Shared object file" }, 97 { ET_CORE, "Core file" }, 98 }; 99 100 static const struct machine_table_t 101 { 102 const Elf64_Half key; 103 const char* str; 104 } machine_table[] = { 105 { EM_NONE, "No machine" }, 106 { EM_M32, "AT&T WE 32100" }, 107 { EM_SPARC, "SUN SPARC" }, 108 { EM_386, "Intel 80386" }, 109 { EM_68K, "Motorola m68k family" }, 110 { EM_88K, "Motorola m88k family" }, 111 { EM_486, "Intel 80486// Reserved for future use" }, 112 { EM_860, "Intel 80860" }, 113 { EM_MIPS, "MIPS R3000 (officially, big-endian only)" }, 114 { EM_S370, "IBM System/370" }, 115 { EM_MIPS_RS3_LE, 116 "MIPS R3000 little-endian (Oct 4 1999 Draft) Deprecated" }, 117 { EM_res011, "Reserved" }, 118 { EM_res012, "Reserved" }, 119 { EM_res013, "Reserved" }, 120 { EM_res014, "Reserved" }, 121 { EM_PARISC, "HPPA" }, 122 { EM_res016, "Reserved" }, 123 { EM_VPP550, "Fujitsu VPP500" }, 124 { EM_SPARC32PLUS, "Sun's v8plus" }, 125 { EM_960, "Intel 80960" }, 126 { EM_PPC, "PowerPC" }, 127 { EM_PPC64, "64-bit PowerPC" }, 128 { EM_S390, "IBM S/390" }, 129 { EM_SPU, "Sony/Toshiba/IBM SPU" }, 130 { EM_res024, "Reserved" }, 131 { EM_res025, "Reserved" }, 132 { EM_res026, "Reserved" }, 133 { EM_res027, "Reserved" }, 134 { EM_res028, "Reserved" }, 135 { EM_res029, "Reserved" }, 136 { EM_res030, "Reserved" }, 137 { EM_res031, "Reserved" }, 138 { EM_res032, "Reserved" }, 139 { EM_res033, "Reserved" }, 140 { EM_res034, "Reserved" }, 141 { EM_res035, "Reserved" }, 142 { EM_V800, "NEC V800 series" }, 143 { EM_FR20, "Fujitsu FR20" }, 144 { EM_RH32, "TRW RH32" }, 145 { EM_MCORE, "Motorola M*Core // May also be taken by Fujitsu MMA" }, 146 { EM_RCE, "Old name for MCore" }, 147 { EM_ARM, "ARM" }, 148 { EM_OLD_ALPHA, "Digital Alpha" }, 149 { EM_SH, "Renesas (formerly Hitachi) / SuperH SH" }, 150 { EM_SPARCV9, "SPARC v9 64-bit" }, 151 { EM_TRICORE, "Siemens Tricore embedded processor" }, 152 { EM_ARC, "ARC Cores" }, 153 { EM_H8_300, "Renesas (formerly Hitachi) H8/300" }, 154 { EM_H8_300H, "Renesas (formerly Hitachi) H8/300H" }, 155 { EM_H8S, "Renesas (formerly Hitachi) H8S" }, 156 { EM_H8_500, "Renesas (formerly Hitachi) H8/500" }, 157 { EM_IA_64, "Intel IA-64 Processor" }, 158 { EM_MIPS_X, "Stanford MIPS-X" }, 159 { EM_COLDFIRE, "Motorola Coldfire" }, 160 { EM_68HC12, "Motorola M68HC12" }, 161 { EM_MMA, "Fujitsu Multimedia Accelerator" }, 162 { EM_PCP, "Siemens PCP" }, 163 { EM_NCPU, "Sony nCPU embedded RISC processor" }, 164 { EM_NDR1, "Denso NDR1 microprocesspr" }, 165 { EM_STARCORE, "Motorola Star*Core processor" }, 166 { EM_ME16, "Toyota ME16 processor" }, 167 { EM_ST100, "STMicroelectronics ST100 processor" }, 168 { EM_TINYJ, "Advanced Logic Corp. TinyJ embedded processor" }, 169 { EM_X86_64, "Advanced Micro Devices X86-64 processor" }, 170 { EM_PDSP, "Sony DSP Processor" }, 171 { EM_PDP10, "Digital Equipment Corp. PDP-10" }, 172 { EM_PDP11, "Digital Equipment Corp. PDP-11" }, 173 { EM_FX66, "Siemens FX66 microcontroller" }, 174 { EM_ST9PLUS, "STMicroelectronics ST9+ 8/16 bit microcontroller" }, 175 { EM_ST7, "STMicroelectronics ST7 8-bit microcontroller" }, 176 { EM_68HC16, "Motorola MC68HC16 Microcontroller" }, 177 { EM_68HC11, "Motorola MC68HC11 Microcontroller" }, 178 { EM_68HC08, "Motorola MC68HC08 Microcontroller" }, 179 { EM_68HC05, "Motorola MC68HC05 Microcontroller" }, 180 { EM_SVX, "Silicon Graphics SVx" }, 181 { EM_ST19, "STMicroelectronics ST19 8-bit cpu" }, 182 { EM_VAX, "Digital VAX" }, 183 { EM_CRIS, "Axis Communications 32-bit embedded processor" }, 184 { EM_JAVELIN, "Infineon Technologies 32-bit embedded cpu" }, 185 { EM_FIREPATH, "Element 14 64-bit DSP processor" }, 186 { EM_ZSP, "LSI Logic's 16-bit DSP processor" }, 187 { EM_MMIX, "Donald Knuth's educational 64-bit processor" }, 188 { EM_HUANY, "Harvard's machine-independent format" }, 189 { EM_PRISM, "SiTera Prism" }, 190 { EM_AVR, "Atmel AVR 8-bit microcontroller" }, 191 { EM_FR30, "Fujitsu FR30" }, 192 { EM_D10V, "Mitsubishi D10V" }, 193 { EM_D30V, "Mitsubishi D30V" }, 194 { EM_V850, "NEC v850" }, 195 { EM_M32R, "Renesas M32R (formerly Mitsubishi M32R)" }, 196 { EM_MN10300, "Matsushita MN10300" }, 197 { EM_MN10200, "Matsushita MN10200" }, 198 { EM_PJ, "picoJava" }, 199 { EM_OPENRISC, "OpenRISC 32-bit embedded processor" }, 200 { EM_ARC_A5, "ARC Cores Tangent-A5" }, 201 { EM_XTENSA, "Tensilica Xtensa Architecture" }, 202 { EM_VIDEOCORE, "Alphamosaic VideoCore processor" }, 203 { EM_TMM_GPP, "Thompson Multimedia General Purpose Processor" }, 204 { EM_NS32K, "National Semiconductor 32000 series" }, 205 { EM_TPC, "Tenor Network TPC processor" }, 206 { EM_SNP1K, "Trebia SNP 1000 processor" }, 207 { EM_ST200, "STMicroelectronics ST200 microcontroller" }, 208 { EM_IP2K, "Ubicom IP2022 micro controller" }, 209 { EM_MAX, "MAX Processor" }, 210 { EM_CR, "National Semiconductor CompactRISC" }, 211 { EM_F2MC16, "Fujitsu F2MC16" }, 212 { EM_MSP430, "TI msp430 micro controller" }, 213 { EM_BLACKFIN, "ADI Blackfin" }, 214 { EM_SE_C33, "S1C33 Family of Seiko Epson processors" }, 215 { EM_SEP, "Sharp embedded microprocessor" }, 216 { EM_ARCA, "Arca RISC Microprocessor" }, 217 { EM_UNICORE, "Microprocessor series from PKU-Unity Ltd. and MPRC of " 218 "Peking University" }, 219 { EM_EXCESS, "eXcess: 16/32/64-bit configurable embedded CPU" }, 220 { EM_DXP, "Icera Semiconductor Inc. Deep Execution Processor" }, 221 { EM_ALTERA_NIOS2, "Altera Nios II soft-core processor" }, 222 { EM_CRX, "National Semiconductor CRX" }, 223 { EM_XGATE, "Motorola XGATE embedded processor" }, 224 { EM_C166, "Infineon C16x/XC16x processor" }, 225 { EM_M16C, "Renesas M16C series microprocessors" }, 226 { EM_DSPIC30F, "Microchip Technology dsPIC30F Digital Signal Controller" }, 227 { EM_CE, "Freescale Communication Engine RISC core" }, 228 { EM_M32C, "Renesas M32C series microprocessors" }, 229 { EM_res121, "Reserved" }, 230 { EM_res122, "Reserved" }, 231 { EM_res123, "Reserved" }, 232 { EM_res124, "Reserved" }, 233 { EM_res125, "Reserved" }, 234 { EM_res126, "Reserved" }, 235 { EM_res127, "Reserved" }, 236 { EM_res128, "Reserved" }, 237 { EM_res129, "Reserved" }, 238 { EM_res130, "Reserved" }, 239 { EM_TSK3000, "Altium TSK3000 core" }, 240 { EM_RS08, "Freescale RS08 embedded processor" }, 241 { EM_res133, "Reserved" }, 242 { EM_ECOG2, "Cyan Technology eCOG2 microprocessor" }, 243 { EM_SCORE, "Sunplus Score" }, 244 { EM_SCORE7, "Sunplus S+core7 RISC processor" }, 245 { EM_DSP24, "New Japan Radio (NJR) 24-bit DSP Processor" }, 246 { EM_VIDEOCORE3, "Broadcom VideoCore III processor" }, 247 { EM_LATTICEMICO32, "RISC processor for Lattice FPGA architecture" }, 248 { EM_SE_C17, "Seiko Epson C17 family" }, 249 { EM_TI_C6000, "Texas Instruments TMS320C6000 DSP family" }, 250 { EM_TI_C2000, "Texas Instruments TMS320C2000 DSP family" }, 251 { EM_TI_C5500, "Texas Instruments TMS320C55x DSP family" }, 252 { EM_res143, "Reserved" }, 253 { EM_res144, "Reserved" }, 254 { EM_res145, "Reserved" }, 255 { EM_res146, "Reserved" }, 256 { EM_res147, "Reserved" }, 257 { EM_res148, "Reserved" }, 258 { EM_res149, "Reserved" }, 259 { EM_res150, "Reserved" }, 260 { EM_res151, "Reserved" }, 261 { EM_res152, "Reserved" }, 262 { EM_res153, "Reserved" }, 263 { EM_res154, "Reserved" }, 264 { EM_res155, "Reserved" }, 265 { EM_res156, "Reserved" }, 266 { EM_res157, "Reserved" }, 267 { EM_res158, "Reserved" }, 268 { EM_res159, "Reserved" }, 269 { EM_MMDSP_PLUS, "STMicroelectronics 64bit VLIW Data Signal Processor" }, 270 { EM_CYPRESS_M8C, "Cypress M8C microprocessor" }, 271 { EM_R32C, "Renesas R32C series microprocessors" }, 272 { EM_TRIMEDIA, "NXP Semiconductors TriMedia architecture family" }, 273 { EM_QDSP6, "QUALCOMM DSP6 Processor" }, 274 { EM_8051, "Intel 8051 and variants" }, 275 { EM_STXP7X, "STMicroelectronics STxP7x family" }, 276 { EM_NDS32, 277 "Andes Technology compact code size embedded RISC processor family" }, 278 { EM_ECOG1, "Cyan Technology eCOG1X family" }, 279 { EM_ECOG1X, "Cyan Technology eCOG1X family" }, 280 { EM_MAXQ30, "Dallas Semiconductor MAXQ30 Core Micro-controllers" }, 281 { EM_XIMO16, "New Japan Radio (NJR) 16-bit DSP Processor" }, 282 { EM_MANIK, "M2000 Reconfigurable RISC Microprocessor" }, 283 { EM_CRAYNV2, "Cray Inc. NV2 vector architecture" }, 284 { EM_RX, "Renesas RX family" }, 285 { EM_METAG, "Imagination Technologies META processor architecture" }, 286 { EM_MCST_ELBRUS, "MCST Elbrus general purpose hardware architecture" }, 287 { EM_ECOG16, "Cyan Technology eCOG16 family" }, 288 { EM_CR16, "National Semiconductor CompactRISC 16-bit processor" }, 289 { EM_ETPU, "Freescale Extended Time Processing Unit" }, 290 { EM_SLE9X, "Infineon Technologies SLE9X core" }, 291 { EM_L1OM, "Intel L1OM" }, 292 { EM_INTEL181, "Reserved by Intel" }, 293 { EM_INTEL182, "Reserved by Intel" }, 294 { EM_AARCH64, "ARM AArch64" }, 295 { EM_res184, "Reserved by ARM" }, 296 { EM_AVR32, "Atmel Corporation 32-bit microprocessor family" }, 297 { EM_STM8, "STMicroeletronics STM8 8-bit microcontroller" }, 298 { EM_TILE64, "Tilera TILE64 multicore architecture family" }, 299 { EM_TILEPRO, "Tilera TILEPro multicore architecture family" }, 300 { EM_MICROBLAZE, "Xilinx MicroBlaze 32-bit RISC soft processor core" }, 301 { EM_CUDA, "NVIDIA CUDA architecture " }, 302 { EM_TILEGX, "Tilera TILE-Gx multicore architecture family" }, 303 { EM_CLOUDSHIELD, "CloudShield architecture family" }, 304 { EM_COREA_1ST, "KIPO-KAIST Core-A 1st generation processor family" }, 305 { EM_COREA_2ND, "KIPO-KAIST Core-A 2nd generation processor family" }, 306 { EM_ARC_COMPACT2, "Synopsys ARCompact V2" }, 307 { EM_OPEN8, "Open8 8-bit RISC soft processor core" }, 308 { EM_RL78, "Renesas RL78 family" }, 309 { EM_VIDEOCORE5, "Broadcom VideoCore V processor" }, 310 { EM_78KOR, "Renesas 78KOR family" }, 311 { EM_56800EX, "Freescale 56800EX Digital Signal Controller (DSC)" }, 312 { EM_BA1, "Beyond BA1 CPU architecture" }, 313 { EM_BA2, "Beyond BA2 CPU architecture" }, 314 { EM_XCORE, "XMOS xCORE processor family" }, 315 { EM_MCHP_PIC, "Microchip 8-bit PIC(r) family" }, 316 { EM_INTEL205, "Reserved by Intel" }, 317 { EM_INTEL206, "Reserved by Intel" }, 318 { EM_INTEL207, "Reserved by Intel" }, 319 { EM_INTEL208, "Reserved by Intel" }, 320 { EM_INTEL209, "Reserved by Intel" }, 321 { EM_KM32, "KM211 KM32 32-bit processor" }, 322 { EM_KMX32, "KM211 KMX32 32-bit processor" }, 323 { EM_KMX16, "KM211 KMX16 16-bit processor" }, 324 { EM_KMX8, "KM211 KMX8 8-bit processor" }, 325 { EM_KVARC, "KM211 KVARC processor" }, 326 { EM_CDP, "Paneve CDP architecture family" }, 327 { EM_COGE, "Cognitive Smart Memory Processor" }, 328 { EM_COOL, "iCelero CoolEngine" }, 329 { EM_NORC, "Nanoradio Optimized RISC" }, 330 { EM_CSR_KALIMBA, "CSR Kalimba architecture family" }, 331 { EM_Z80, "Zilog Z80" }, 332 { EM_VISIUM, "Controls and Data Services VISIUMcore processor" }, 333 { EM_FT32, "FTDI Chip FT32 high performance 32-bit RISC architecture" }, 334 { EM_MOXIE, "Moxie processor family" }, 335 { EM_AMDGPU, "AMD GPU architecture" }, 336 { EM_RISCV, "RISC-V" }, 337 { EM_LANAI, "Lanai processor" }, 338 { EM_CEVA, "CEVA Processor Architecture Family" }, 339 { EM_CEVA_X2, "CEVA X2 Processor Family" }, 340 { EM_BPF, "Linux BPF – in-kernel virtual machine" }, 341 { EM_GRAPHCORE_IPU, "Graphcore Intelligent Processing Unit" }, 342 { EM_IMG1, "Imagination Technologies" }, 343 { EM_NFP, "Netronome Flow Processor (P)" }, 344 { EM_CSKY, "C-SKY processor family" }, 345 { EM_ARC_COMPACT3_64, "Synopsys ARCv2.3 64-bit" }, 346 { EM_MCS6502, "MOS Technology MCS 6502 processor" }, 347 { EM_ARC_COMPACT3, "Synopsys ARCv2.3 32-bit" }, 348 { EM_KVX, "Kalray VLIW core of the MPPA processor family" }, 349 { EM_65816, "WDC 65816/65C816" }, 350 { EM_LOONGARCH, "Loongson Loongarch" }, 351 { EM_KF32, "ChipON KungFu32" }, 352 { EM_MT, "Morpho Techologies MT processor" }, 353 { EM_ALPHA, "Alpha" }, 354 { EM_WEBASSEMBLY, "Web Assembly" }, 355 { EM_DLX, "OpenDLX" }, 356 { EM_XSTORMY16, "Sanyo XStormy16 CPU core" }, 357 { EM_IQ2000, "Vitesse IQ2000" }, 358 { EM_M32C_OLD, "M32C_OLD" }, 359 { EM_NIOS32, "Altera Nios" }, 360 { EM_CYGNUS_MEP, "Toshiba MeP Media Engine" }, 361 { EM_ADAPTEVA_EPIPHANY, "Adapteva EPIPHANY" }, 362 { EM_CYGNUS_FRV, "Fujitsu FR-V" }, 363 { EM_S12Z, "Freescale S12Z" }, 364 }; 365 366 static const struct section_type_table_t 367 { 368 const Elf64_Word key; 369 const char* str; 370 } section_type_table[] = { 371 { SHT_NULL, "NULL" }, 372 { SHT_PROGBITS, "PROGBITS" }, 373 { SHT_SYMTAB, "SYMTAB" }, 374 { SHT_STRTAB, "STRTAB" }, 375 { SHT_RELA, "RELA" }, 376 { SHT_HASH, "HASH" }, 377 { SHT_DYNAMIC, "DYNAMIC" }, 378 { SHT_NOTE, "NOTE" }, 379 { SHT_NOBITS, "NOBITS" }, 380 { SHT_REL, "REL" }, 381 { SHT_SHLIB, "SHLIB" }, 382 { SHT_DYNSYM, "DYNSYM" }, 383 { SHT_INIT_ARRAY, "INIT_ARRAY" }, 384 { SHT_FINI_ARRAY, "FINI_ARRAY" }, 385 { SHT_PREINIT_ARRAY, "PREINIT_ARRAY" }, 386 { SHT_GROUP, "GROUP" }, 387 { SHT_SYMTAB_SHNDX, "SYMTAB_SHNDX" }, 388 { SHT_GNU_ATTRIBUTES, "GNU_ATTRIBUTES" }, 389 { SHT_GNU_HASH, "GNU_HASH" }, 390 { SHT_GNU_LIBLIST, "GNU_LIBLIST" }, 391 { SHT_CHECKSUM, "CHECKSUM" }, 392 { SHT_LOSUNW, "LOSUNW" }, 393 { SHT_SUNW_move, "SUNW_move" }, 394 { SHT_SUNW_COMDAT, "SUNW_COMDAT" }, 395 { SHT_SUNW_syminfo, "SUNW_syminfo" }, 396 { SHT_GNU_verdef, "GNU_verdef" }, 397 { SHT_GNU_verneed, "GNU_verneed" }, 398 { SHT_GNU_versym, "GNU_versym" }, 399 { SHT_ARM_EXIDX, "ARM_EXIDX" }, 400 { SHT_ARM_PREEMPTMAP, "ARM_PREEMPTMAP" }, 401 { SHT_ARM_ATTRIBUTES, "ARM_ATTRIBUTES" }, 402 { SHT_ARM_DEBUGOVERLAY, "ARM_DEBUGOVERLAY" }, 403 { SHT_ARM_OVERLAYSECTION, "ARM_OVERLAYSECTION" }, 404 405 }; 406 407 static const struct segment_type_table_t 408 { 409 const Elf_Word key; 410 const char* str; 411 } segment_type_table[] = { 412 { PT_NULL, "NULL" }, 413 { PT_LOAD, "LOAD" }, 414 { PT_DYNAMIC, "DYNAMIC" }, 415 { PT_INTERP, "INTERP" }, 416 { PT_NOTE, "NOTE" }, 417 { PT_SHLIB, "SHLIB" }, 418 { PT_PHDR, "PHDR" }, 419 { PT_TLS, "TLS" }, 420 { PT_GNU_EH_FRAME, "GNU_EH_FRAME" }, 421 { PT_GNU_STACK, "GNU_STACK" }, 422 { PT_GNU_RELRO, "GNU_RELRO" }, 423 { PT_GNU_PROPERTY, "GNU_PROPERTY" }, 424 { PT_GNU_MBIND_LO, "GNU_MBIND_LO" }, 425 { PT_GNU_MBIND_HI, "GNU_MBIND_HI" }, 426 { PT_PAX_FLAGS, "PAX_FLAGS" }, 427 { PT_OPENBSD_RANDOMIZE, "OPENBSD_RANDOMIZE" }, 428 { PT_OPENBSD_WXNEEDED, "OPENBSD_WXNEEDED " }, 429 { PT_OPENBSD_BOOTDATA, "OPENBSD_BOOTDATA " }, 430 { PT_SUNWBSS, "PT_SUNWBSS" }, 431 { PT_SUNWSTACK, "SUNWSTACK" }, 432 }; 433 434 static const struct segment_flag_table_t 435 { 436 const Elf_Word key; 437 const char* str; 438 } segment_flag_table[] = { 439 { 0, " " }, { 1, " E" }, { 2, " W " }, { 3, " WE" }, 440 { 4, "R " }, { 5, "R E" }, { 6, "RW " }, { 7, "RWE" }, 441 }; 442 443 static const struct symbol_bind_t 444 { 445 const Elf_Word key; 446 const char* str; 447 } symbol_bind_table[] = { 448 { STB_LOCAL, "LOCAL" }, { STB_GLOBAL, "GLOBAL" }, 449 { STB_WEAK, "WEAK" }, { STB_LOOS, "LOOS" }, 450 { STB_HIOS, "HIOS" }, { STB_MULTIDEF, "MULTIDEF" }, 451 { STB_LOPROC, "LOPROC" }, { STB_HIPROC, "HIPROC" }, 452 }; 453 454 static const struct symbol_type_t 455 { 456 const Elf_Word key; 457 const char* str; 458 } symbol_type_table[] = { 459 { STT_NOTYPE, "NOTYPE" }, { STT_OBJECT, "OBJECT" }, 460 { STT_FUNC, "FUNC" }, { STT_SECTION, "SECTION" }, 461 { STT_FILE, "FILE" }, { STT_COMMON, "COMMON" }, 462 { STT_TLS, "TLS" }, { STT_LOOS, "LOOS" }, 463 { STT_HIOS, "HIOS" }, { STT_LOPROC, "LOPROC" }, 464 { STT_HIPROC, "HIPROC" }, 465 }; 466 467 static const struct dynamic_tag_t 468 { 469 const Elf_Word key; 470 const char* str; 471 } dynamic_tag_table[] = { 472 { DT_NULL, "NULL" }, 473 { DT_NEEDED, "NEEDED" }, 474 { DT_PLTRELSZ, "PLTRELSZ" }, 475 { DT_PLTGOT, "PLTGOT" }, 476 { DT_HASH, "HASH" }, 477 { DT_STRTAB, "STRTAB" }, 478 { DT_SYMTAB, "SYMTAB" }, 479 { DT_RELA, "RELA" }, 480 { DT_RELASZ, "RELASZ" }, 481 { DT_RELAENT, "RELAENT" }, 482 { DT_STRSZ, "STRSZ" }, 483 { DT_SYMENT, "SYMENT" }, 484 { DT_INIT, "INIT" }, 485 { DT_FINI, "FINI" }, 486 { DT_SONAME, "SONAME" }, 487 { DT_RPATH, "RPATH" }, 488 { DT_SYMBOLIC, "SYMBOLIC" }, 489 { DT_REL, "REL" }, 490 { DT_RELSZ, "RELSZ" }, 491 { DT_RELENT, "RELENT" }, 492 { DT_PLTREL, "PLTREL" }, 493 { DT_DEBUG, "DEBUG" }, 494 { DT_TEXTREL, "TEXTREL" }, 495 { DT_JMPREL, "JMPREL" }, 496 { DT_BIND_NOW, "BIND_NOW" }, 497 { DT_INIT_ARRAY, "INIT_ARRAY" }, 498 { DT_FINI_ARRAY, "FINI_ARRAY" }, 499 { DT_INIT_ARRAYSZ, "INIT_ARRAYSZ" }, 500 { DT_FINI_ARRAYSZ, "FINI_ARRAYSZ" }, 501 { DT_RUNPATH, "RUNPATH" }, 502 { DT_FLAGS, "FLAGS" }, 503 { DT_ENCODING, "ENCODING" }, 504 { DT_PREINIT_ARRAY, "PREINIT_ARRAY" }, 505 { DT_PREINIT_ARRAYSZ, "PREINIT_ARRAYSZ" }, 506 { DT_MAXPOSTAGS, "MAXPOSTAGS" }, 507 { DT_GNU_HASH, "GNU_HASH" }, 508 { DT_TLSDESC_PLT, "TLSDESC_PLT" }, 509 { DT_TLSDESC_GOT, "TLSDESC_GOT" }, 510 { DT_GNU_CONFLICT, "GNU_CONFLICT" }, 511 { DT_GNU_LIBLIST, "GNU_LIBLIST" }, 512 { DT_CONFIG, "CONFIG" }, 513 { DT_DEPAUDIT, "DEPAUDIT" }, 514 { DT_AUDIT, "AUDIT" }, 515 { DT_PLTPAD, "PLTPAD" }, 516 { DT_MOVETAB, "MOVETAB" }, 517 { DT_SYMINFO, "SYMINFO" }, 518 { DT_ADDRRNGHI, "ADDRRNGHI" }, 519 { DT_VERSYM, "VERSYM" }, 520 { DT_RELACOUNT, "RELACOUNT" }, 521 { DT_RELCOUNT, "RELCOUNT" }, 522 { DT_FLAGS_1, "FLAGS_1" }, 523 { DT_VERDEF, "VERDEF" }, 524 { DT_VERDEFNUM, "VERDEFNUM" }, 525 { DT_VERNEED, "VERNEED" }, 526 { DT_VERNEEDNUM, "VERNEEDNUM" }, 527 }; 528 529 // clang-format off 530 static const struct note_tag_t 531 { 532 struct note_values_t 533 { 534 Elf64_Word type; 535 std::string type_str; 536 std::string description; 537 }; 538 std::string name; 539 std::vector<note_values_t> values; 540 } note_tag_table[] = { 541 { "", 542 { { NT_PRSTATUS, "NT_PRSTATUS", "prstatus struct" }, 543 { NT_FPREGSET, "NT_FPREGSET", "fpregset struct" }, 544 { NT_PRPSINFO, "NT_PRPSINFO", "prpsinfo struct" }, 545 { NT_TASKSTRUCT, "NT_TASKSTRUCT", "task struct" }, 546 { NT_AUXV, "NT_AUXV", "Elfxx_auxv_t" }, 547 { NT_PSTATUS, "NT_PSTATUS", "pstatus struct" }, 548 { NT_FPREGS, "NT_FPREGS", "fpregset struct" }, 549 { NT_PSINFO, "NT_PSINFO", "psinfo struct" }, 550 { NT_LWPSTATUS, "NT_LWPSTATUS", "lwpstatus_t struct" }, 551 { NT_LWPSINFO, "NT_LWPSINFO", "lwpsinfo_t struct" }, 552 { NT_WIN32PSTATUS, "NT_WIN32PSTATUS", "win32_pstatus struct" }, 553 } }, 554 { "LINUX", 555 { { NT_PRXFPREG, "NT_PRXFPREG", "Contains a user_xfpregs_struct;" }, 556 { NT_PPC_VMX, "NT_PPC_VMX", "PowerPC Altivec/VMX registers" }, 557 { NT_PPC_VSX, "NT_PPC_VSX", "PowerPC VSX registers" }, 558 { NT_PPC_TAR, "NT_PPC_TAR", "PowerPC Target Address Register" }, 559 { NT_PPC_PPR, "NT_PPC_PPR", "PowerPC Program Priority Register" }, 560 { NT_PPC_DSCR, "NT_PPC_DSCR", "PowerPC Data Stream Control Register" }, 561 { NT_PPC_EBB, "NT_PPC_EBB", "PowerPC Event Based Branch Registers" }, 562 { NT_PPC_PMU, "NT_PPC_PMU", "PowerPC Performance Monitor Registers" }, 563 { NT_PPC_TM_CGPR, "NT_PPC_TM_CGPR", "PowerPC TM checkpointed GPR Registers" }, 564 { NT_PPC_TM_CFPR, "NT_PPC_TM_CFPR", "PowerPC TM checkpointed FPR Registers" }, 565 { NT_PPC_TM_CVMX, "NT_PPC_TM_CVMX", "PowerPC TM checkpointed VMX Registers" }, 566 { NT_PPC_TM_CVSX, "NT_PPC_TM_CVSX", "PowerPC TM checkpointed VSX Registers" }, 567 { NT_PPC_TM_SPR, "NT_PPC_TM_SPR", "PowerPC TM Special Purpose Registers" }, 568 { NT_PPC_TM_CTAR, "NT_PPC_TM_CTAR", "PowerPC TM checkpointed TAR" }, 569 { NT_PPC_TM_CPPR, "NT_PPC_TM_CPPR", "PowerPC TM checkpointed PPR" }, 570 { NT_PPC_TM_CDSCR, "NT_PPC_TM_CDSCR", "PowerPC TM checkpointed Data SCR" }, 571 { NT_386_TLS, "NT_386_TLS", "x86 TLS information" }, 572 { NT_386_IOPERM, "NT_386_IOPERM", "x86 io permissions" }, 573 { NT_X86_XSTATE, "NT_X86_XSTATE", "x86 XSAVE extended state" }, 574 { NT_X86_CET, "NT_X86_CET", "x86 CET state" }, 575 { NT_S390_HIGH_GPRS, "NT_S390_HIGH_GPRS", "S/390 upper halves of GPRs " }, 576 { NT_S390_TIMER, "NT_S390_TIMER", "S390 timer" }, 577 { NT_S390_TODCMP, "NT_S390_TODCMP", "S390 TOD clock comparator" }, 578 { NT_S390_TODPREG, "NT_S390_TODPREG", "S390 TOD programmable register" }, 579 { NT_S390_CTRS, "NT_S390_CTRS", "S390 control registers" }, 580 { NT_S390_PREFIX, "NT_S390_PREFIX", "S390 prefix register" }, 581 { NT_S390_LAST_BREAK, "NT_S390_LAST_BREAK", "S390 breaking event address" }, 582 { NT_S390_SYSTEM_CALL, "NT_S390_SYSTEM_CALL", "S390 system call restart data" }, 583 { NT_S390_TDB, "NT_S390_TDB", "S390 transaction diagnostic block" }, 584 { NT_S390_VXRS_LOW, "NT_S390_VXRS_LOW", "S390 vector registers 0-15 upper half" }, 585 { NT_S390_VXRS_HIGH, "NT_S390_VXRS_HIGH", "S390 vector registers 16-31" }, 586 { NT_S390_GS_CB, "NT_S390_GS_CB", "s390 guarded storage registers" }, 587 { NT_S390_GS_BC, "NT_S390_GS_BC", "s390 guarded storage broadcast control block" }, 588 { NT_ARM_VFP, "NT_ARM_VFP", "ARM VFP registers" }, 589 { NT_ARM_TLS, "NT_ARM_TLS", "AArch TLS registers" }, 590 { NT_ARM_HW_BREAK, "NT_ARM_HW_BREAK", "AArch hardware breakpoint registers" }, 591 { NT_ARM_HW_WATCH, "NT_ARM_HW_WATCH", "AArch hardware watchpoint registers" }, 592 { NT_ARM_SVE, "NT_ARM_SVE", "AArch SVE registers. " }, 593 { NT_ARM_PAC_MASK, "NT_ARM_PAC_MASK", "AArch pointer authentication code masks" }, 594 { NT_ARM_PACA_KEYS, "NT_ARM_PACA_KEYS", "ARM pointer authentication address keys" }, 595 { NT_ARM_PACG_KEYS, "NT_ARM_PACG_KEYS", "ARM pointer authentication generic keys" }, 596 { NT_ARM_TAGGED_ADDR_CTRL, "NT_ARM_TAGGED_ADDR_CTRL", "AArch64 tagged address control (prctl())" }, 597 { NT_ARM_PAC_ENABLED_KEYS, "NT_ARM_PAC_ENABLED_KEYS", "AArch64 pointer authentication enabled keys (prctl())" }, 598 { NT_ARC_V2, "NT_ARC_V2", "ARC HS accumulator/extra registers. " }, 599 { NT_LARCH_CPUCFG, "NT_LARCH_CPUCFG", "LoongArch CPU config registers" }, 600 { NT_LARCH_CSR, "NT_LARCH_CSR", "LoongArch Control State Registers" }, 601 { NT_LARCH_LSX, "NT_LARCH_LSX", "LoongArch SIMD eXtension registers" }, 602 { NT_LARCH_LASX, "NT_LARCH_LASX", "LoongArch Advanced SIMD eXtension registers" }, 603 { NT_RISCV_CSR, "NT_RISCV_CSR", "RISC-V Control and Status Registers" }, 604 } }, 605 { "CORE", 606 { { NT_LARCH_LBT, "NT_LARCH_LBT", "LoongArch Binary Translation registers" } 607 } }, 608 { "FreeBSD", 609 { { NT_FREEBSD_THRMISC, "NT_FREEBSD_THRMISC", "Thread miscellaneous info." }, 610 { NT_FREEBSD_PROCSTAT_PROC, "NT_FREEBSD_PROCSTAT_PROC", "Procstat proc data." }, 611 { NT_FREEBSD_PROCSTAT_FILES, "NT_FREEBSD_PROCSTAT_FILES", "Procstat files data." }, 612 { NT_FREEBSD_PROCSTAT_VMMAP, "NT_FREEBSD_PROCSTAT_VMMAP", "Procstat vmmap data." }, 613 { NT_FREEBSD_PROCSTAT_GROUPS, "NT_FREEBSD_PROCSTAT_GROUPS", "Procstat groups data." }, 614 { NT_FREEBSD_PROCSTAT_UMASK, "NT_FREEBSD_PROCSTAT_UMASK", "Procstat umask data." }, 615 { NT_FREEBSD_PROCSTAT_RLIMIT, "NT_FREEBSD_PROCSTAT_RLIMIT", "Procstat rlimit data." }, 616 { NT_FREEBSD_PROCSTAT_OSREL, "NT_FREEBSD_PROCSTAT_OSREL", "Procstat osreldate data." }, 617 { NT_FREEBSD_PROCSTAT_PSSTRINGS, "NT_FREEBSD_PROCSTAT_PSSTRINGS", "Procstat ps_strings data." }, 618 { NT_FREEBSD_PROCSTAT_AUXV, "NT_FREEBSD_PROCSTAT_AUXV", "Procstat auxv data." }, 619 { NT_FREEBSD_PTLWPINFO, "NT_FREEBSD_PTLWPINFO", "Thread ptrace miscellaneous info." }, 620 } }, 621 { "NetBSD-CORE", 622 { { NT_NETBSDCORE_PROCINFO, "NT_NETBSDCORE_PROCINFO", "Has a struct procinfo" }, 623 { NT_NETBSDCORE_AUXV, "NT_NETBSDCORE_AUXV", "Has auxv data" }, 624 { NT_NETBSDCORE_LWPSTATUS, "NT_NETBSDCORE_LWPSTATUS", "Has LWPSTATUS data" }, 625 { NT_NETBSDCORE_FIRSTMACH, "NT_NETBSDCORE_FIRSTMACH", "start of machdep note types" }, 626 } }, 627 { "OpenBSD", 628 { { NT_OPENBSD_PROCINFO, "NT_OPENBSD_PROCINFO", "" }, 629 { NT_OPENBSD_AUXV, "NT_OPENBSD_AUXV", "" }, 630 { NT_OPENBSD_REGS, "NT_OPENBSD_REGS", "" }, 631 { NT_OPENBSD_FPREGS, "NT_OPENBSD_FPREGS", "" }, 632 { NT_OPENBSD_XFPREGS, "NT_OPENBSD_XFPREGS", "" }, 633 { NT_OPENBSD_WCOOKIE, "NT_OPENBSD_WCOOKIE", "" }, 634 } }, 635 { "SPU", 636 { { NT_SPU, "NT_SPU", "" } 637 } }, 638 { "GNU", 639 { 640 { NT_GNU_ABI_TAG, "NT_GNU_ABI_TAG", "GNU ABI tag" }, 641 { NT_GNU_HWCAP, "NT_GNU_HWCAP", "Used by ld.so and kernel vDSO" }, 642 { NT_GNU_BUILD_ID, "NT_GNU_BUILD_ID", "Build ID of the binary" }, 643 { NT_GNU_GOLD_VERSION, "NT_GNU_GOLD_VERSION", "Version of GNU gold used to link the binary" }, 644 { NT_GNU_PROPERTY_TYPE_0, "NT_GNU_PROPERTY_TYPE_0", "Property type 0" }, 645 // { NT_GNU_PROPERTY_TYPE_1, "NT_GNU_PROPERTY_TYPE_1", "Property type 1" }, 646 // { NT_GNU_PROPERTY_TYPE_2, "NT_GNU_PROPERTY_TYPE_2", "Property type 2" }, 647 // { NT_GNU_PROPERTY_TYPE_3, "NT_GNU_PROPERTY_TYPE_3", "Property type 3" }, 648 // { NT_GNU_PROPERTY_TYPE_4, "NT_GNU_PROPERTY_TYPE_4", "Property type 4" }, 649 // { NT_GNU_PROPERTY_TYPE_5, "NT_GNU_PROPERTY_TYPE_5", "Property type 5" }, 650 // { NT_GNU_PROPERTY_TYPE_6, "NT_GNU_PROPERTY_TYPE_6", "Property type 6" }, 651 // { NT_GNU_PROPERTY_TYPE_7, "NT_GNU_PROPERTY_TYPE_7", "Property type 7" }, 652 // { NT_GNU_PROPERTY_TYPE_8, "NT_GNU_PROPERTY_TYPE_8", "Property type 8" }, 653 // { NT_GNU_PROPERTY_TYPE_9, "NT_GNU_PROPERTY_TYPE_9", "Property type 9" }, 654 // { NT_GNU_PROPERTY_TYPE_10, "NT_GNU_PROPERTY_TYPE_10", "Property type 10" }, 655 // { NT_GNU_PROPERTY_TYPE_11, "NT_GNU_PROPERTY_TYPE_11", "Property type 11" }, 656 // { NT_GNU_PROPERTY_TYPE_12, "NT_GNU_PROPERTY_TYPE_12", "Property type 12" }, 657 // { NT_GNU_PROPERTY_TYPE_13, "NT_GNU_PROPERTY_TYPE_13", "Property type 13" }, 658 // { NT_GNU_PROPERTY_TYPE_14, "NT_GNU_PROPERTY_TYPE_14", "Property type 14" }, 659 } }, 660 // { "SOLARIS", 661 // { { NT_SOLARIS_AUXV, "NT_SOLARIS_AUXV", "" } 662 // } }, 663 // { "AIX", 664 // { { NT_AIX_AUXV, "NT_AIX_AUXV", "" } 665 // } }, 666 // { "IRIX", 667 // { { NT_IRIX_FPREGS, "NT_IRIX_FPREGS", "" } 668 // } }, 669 }; 670 // clang-format on 671 672 static const ELFIO::Elf_Xword MAX_DATA_ENTRIES = 64; 673 674 //------------------------------------------------------------------------------ 675 class dump 676 { 677 #define DUMP_DEC_FORMAT( width ) \ 678 std::setw( width ) << std::setfill( ' ' ) << std::dec << std::right 679 #define DUMP_HEX0x_FORMAT( width ) \ 680 "0x" << std::setw( width ) << std::setfill( '0' ) << std::hex << std::right 681 #define DUMP_HEX_FORMAT( width ) \ 682 std::setw( width ) << std::setfill( '0' ) << std::hex << std::right 683 #define DUMP_STR_FORMAT( width ) \ 684 std::setw( width ) << std::setfill( ' ' ) << std::hex << std::left 685 686 public: 687 //------------------------------------------------------------------------------ header(std::ostream & out,const elfio & reader)688 static void header( std::ostream& out, const elfio& reader ) 689 { 690 if ( !reader.get_header_size() ) { 691 return; 692 } 693 out << "ELF Header" << std::endl 694 << std::endl 695 << " Class: " << str_class( reader.get_class() ) << std::endl 696 << " Encoding: " << str_endian( reader.get_encoding() ) 697 << std::endl 698 << " ELFVersion: " << str_version( reader.get_elf_version() ) 699 << std::endl 700 << " OS/ABI: " << str_os_abi( reader.get_os_abi() ) 701 << std::endl 702 << " ABI Version:" << (int)reader.get_abi_version() << std::endl 703 << " Type: " << str_type( reader.get_type() ) << std::endl 704 << " Machine: " << str_machine( reader.get_machine() ) 705 << std::endl 706 << " Version: " << str_version( reader.get_version() ) 707 << std::endl 708 << " Entry: " 709 << "0x" << std::hex << reader.get_entry() << std::endl 710 << " Flags: " 711 << "0x" << std::hex << reader.get_flags() << std::endl 712 << std::endl; 713 } 714 715 //------------------------------------------------------------------------------ section_headers(std::ostream & out,const elfio & reader)716 static void section_headers( std::ostream& out, const elfio& reader ) 717 { 718 Elf_Half n = reader.sections.size(); 719 720 if ( n == 0 ) { 721 return; 722 } 723 724 out << "Section Headers:" << std::endl; 725 if ( reader.get_class() == ELFCLASS32 ) { // Output for 32-bit 726 out << "[ Nr ] Type Addr Size ES Flg " 727 "Lk Inf Al Name" 728 << std::endl; 729 } 730 else { // Output for 64-bit 731 out << "[ Nr ] Type Addr Size " 732 " Offset Flg" 733 << std::endl 734 << " ES Lk Inf Al Name" << std::endl; 735 } 736 737 for ( Elf_Half i = 0; i < n; ++i ) { // For all sections 738 const section* sec = reader.sections[i]; 739 section_header( out, i, sec, reader.get_class() ); 740 } 741 742 out << "Key to Flags: W (write), A (alloc), X (execute), " << std::endl; 743 out << " M (merge), S (strings), I (info)," << std::endl; 744 out << " L (link order), O (extra OS processing required)," 745 << std::endl; 746 out << " G (group), T (TLS), C (compressed), E (exclude)" 747 << std::endl; 748 } 749 750 //------------------------------------------------------------------------------ section_header(std::ostream & out,Elf_Half no,const section * sec,unsigned char elf_class)751 static void section_header( std::ostream& out, 752 Elf_Half no, 753 const section* sec, 754 unsigned char elf_class ) 755 { 756 std::ios_base::fmtflags original_flags = out.flags(); 757 758 // clang-format off 759 if ( elf_class == ELFCLASS32 ) { // Output for 32-bit 760 out << "[" << DUMP_DEC_FORMAT( 5 ) << no << "] " 761 << DUMP_STR_FORMAT( 17 ) << str_section_type( sec->get_type() ) 762 << " " << DUMP_HEX0x_FORMAT( 8 ) << sec->get_address() << " " 763 << DUMP_HEX0x_FORMAT( 8 ) << sec->get_size() << " " 764 << DUMP_HEX0x_FORMAT( 2 ) << sec->get_entry_size() << " " 765 << DUMP_STR_FORMAT( 3 ) << section_flags( sec->get_flags() ) 766 << " " << DUMP_HEX0x_FORMAT( 2 ) << sec->get_link() << " " 767 << DUMP_HEX0x_FORMAT( 3 ) << sec->get_info() << " " 768 << DUMP_HEX0x_FORMAT( 2 ) << sec->get_addr_align() << " " 769 << DUMP_STR_FORMAT( 17 ) << sec->get_name() << " " << std::endl; 770 } 771 else { // Output for 64-bit 772 out << "[" << DUMP_DEC_FORMAT( 5 ) << no << "] " 773 << DUMP_STR_FORMAT( 17 ) << str_section_type( sec->get_type() ) << " " 774 << DUMP_HEX0x_FORMAT( 16 ) << sec->get_address() << " " 775 << DUMP_HEX0x_FORMAT( 16 ) << sec->get_size() << " " 776 << DUMP_HEX0x_FORMAT( 8 ) << sec->get_offset() << " " 777 << DUMP_STR_FORMAT( 3) << section_flags( sec->get_flags() ) 778 << std::endl 779 << DUMP_STR_FORMAT( 8 ) << " " 780 << DUMP_HEX0x_FORMAT( 4 ) << sec->get_entry_size() << " " 781 << DUMP_HEX0x_FORMAT( 4 ) << sec->get_link() << " " 782 << DUMP_HEX0x_FORMAT( 4 ) << sec->get_info() << " " 783 << DUMP_HEX0x_FORMAT( 4 ) << sec->get_addr_align() << " " 784 << DUMP_STR_FORMAT( 17 ) << sec->get_name() 785 << std::endl; 786 } 787 // clang-format on 788 789 out.flags( original_flags ); 790 791 return; 792 } 793 794 //------------------------------------------------------------------------------ segment_headers(std::ostream & out,const elfio & reader)795 static void segment_headers( std::ostream& out, const elfio& reader ) 796 { 797 Elf_Half n = reader.segments.size(); 798 if ( n == 0 ) { 799 return; 800 } 801 802 out << "Program Headers:" << std::endl; 803 if ( reader.get_class() == ELFCLASS32 ) { // Output for 32-bit 804 out << "[ Nr ] Type VirtAddr PhysAddr FileSize " 805 "Mem.Size Flags Align" 806 << std::endl; 807 } 808 else { // Output for 64-bit 809 out << "[ Nr ] Type Offset VirtAddr " 810 " PhysAddr" 811 812 << std::endl 813 << " FileSize MemSize " 814 " Flags Align" 815 << std::endl; 816 } 817 818 for ( Elf_Half i = 0; i < n; ++i ) { 819 const segment* seg = reader.segments[i]; 820 segment_header( out, i, seg, reader.get_class() ); 821 } 822 823 out << std::endl; 824 } 825 826 //------------------------------------------------------------------------------ segment_header(std::ostream & out,Elf_Half no,const segment * seg,unsigned int elf_class)827 static void segment_header( std::ostream& out, 828 Elf_Half no, 829 const segment* seg, 830 unsigned int elf_class ) 831 { 832 std::ios_base::fmtflags original_flags = out.flags(); 833 // clang-format off 834 if ( elf_class == ELFCLASS32 ) { // Output for 32-bit 835 out << "[" << DUMP_DEC_FORMAT( 5 ) << no << "] " 836 << DUMP_STR_FORMAT( 14 ) << str_segment_type( seg->get_type() ) 837 << " " << DUMP_HEX0x_FORMAT( 8 ) << seg->get_virtual_address() 838 << " " << DUMP_HEX0x_FORMAT( 8 ) << seg->get_physical_address() 839 << " " << DUMP_HEX0x_FORMAT( 8 ) << seg->get_file_size() << " " 840 << DUMP_HEX0x_FORMAT( 8 ) << seg->get_memory_size() << " " 841 << DUMP_STR_FORMAT( 8 ) << str_segment_flag( seg->get_flags() ) 842 << " " << DUMP_HEX0x_FORMAT( 8 ) << seg->get_align() << " " 843 << std::endl; 844 } 845 else { // Output for 64-bit 846 out << "[" << DUMP_DEC_FORMAT( 5 ) << no << "] " 847 << DUMP_STR_FORMAT( 14 ) << str_segment_type( seg->get_type() ) << " " 848 << DUMP_HEX0x_FORMAT( 16 ) << seg->get_offset() << " " 849 << DUMP_HEX0x_FORMAT( 16 ) << seg->get_virtual_address() << " " 850 << DUMP_HEX0x_FORMAT( 16 ) << seg->get_physical_address() 851 << std::endl 852 << DUMP_STR_FORMAT( 23 ) << " " 853 << DUMP_HEX0x_FORMAT( 16 ) << seg->get_file_size() << " " 854 << DUMP_HEX0x_FORMAT( 16 ) << seg->get_memory_size() << " " 855 << DUMP_STR_FORMAT( 3 ) << str_segment_flag( seg->get_flags() ) << " " 856 << DUMP_HEX0x_FORMAT( 1 ) << seg->get_align() 857 << std::endl; 858 } 859 // clang-format on 860 861 out.flags( original_flags ); 862 } 863 864 //------------------------------------------------------------------------------ symbol_tables(std::ostream & out,const elfio & reader)865 static void symbol_tables( std::ostream& out, const elfio& reader ) 866 { 867 for ( const auto& sec : reader.sections ) { // For all sections 868 if ( SHT_SYMTAB == sec->get_type() || 869 SHT_DYNSYM == sec->get_type() ) { 870 const_symbol_section_accessor symbols( reader, sec.get() ); 871 872 Elf_Xword sym_no = symbols.get_symbols_num(); 873 if ( sym_no == 0 ) { 874 continue; 875 } 876 877 out << "Symbol table (" << sec->get_name() << ")" << std::endl; 878 if ( reader.get_class() == ELFCLASS32 ) { // Output for 32-bit 879 out << "[ Nr ] Value Size Type Bind " 880 " Sect Name" 881 << std::endl; 882 } 883 else { // Output for 64-bit 884 out << "[ Nr ] Value Size " 885 "Type Bind Sect" 886 << std::endl 887 << " Name" << std::endl; 888 } 889 for ( Elf_Xword i = 0; i < sym_no; ++i ) { 890 std::string name; 891 Elf64_Addr value = 0; 892 Elf_Xword size = 0; 893 unsigned char bind = 0; 894 unsigned char type = 0; 895 Elf_Half section = 0; 896 unsigned char other = 0; 897 symbols.get_symbol( i, name, value, size, bind, type, 898 section, other ); 899 symbol_table( out, i, name, value, size, bind, type, 900 section, reader.get_class() ); 901 } 902 903 out << std::endl; 904 } 905 } 906 } 907 908 //------------------------------------------------------------------------------ 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)909 static void symbol_table( std::ostream& out, 910 Elf_Xword no, 911 const std::string& name, 912 Elf64_Addr value, 913 Elf_Xword size, 914 unsigned char bind, 915 unsigned char type, 916 Elf_Half section, 917 unsigned int elf_class ) 918 { 919 std::ios_base::fmtflags original_flags = out.flags(); 920 921 if ( elf_class == ELFCLASS32 ) { // Output for 32-bit 922 out << "[" << DUMP_DEC_FORMAT( 5 ) << no << "] " 923 << DUMP_HEX0x_FORMAT( 8 ) << value << " " 924 << DUMP_HEX0x_FORMAT( 8 ) << size << " " << DUMP_STR_FORMAT( 7 ) 925 << str_symbol_type( type ) << " " << DUMP_STR_FORMAT( 8 ) 926 << str_symbol_bind( bind ) << " " << DUMP_DEC_FORMAT( 5 ) 927 << section << " " << DUMP_STR_FORMAT( 1 ) << name << " " 928 << std::endl; 929 } 930 else { // Output for 64-bit 931 out << "[" << DUMP_DEC_FORMAT( 5 ) << no << "] " 932 << DUMP_HEX0x_FORMAT( 16 ) << value << " " 933 << DUMP_HEX0x_FORMAT( 16 ) << size << " " 934 << DUMP_STR_FORMAT( 7 ) << str_symbol_type( type ) << " " 935 << DUMP_STR_FORMAT( 8 ) << str_symbol_bind( bind ) << " " 936 << DUMP_DEC_FORMAT( 5 ) << section << " " << std::endl 937 << " " << DUMP_STR_FORMAT( 1 ) << name << " " 938 << std::endl; 939 } 940 941 out.flags( original_flags ); 942 } 943 944 //------------------------------------------------------------------------------ notes(std::ostream & out,const elfio & reader)945 static void notes( std::ostream& out, const elfio& reader ) 946 { 947 for ( const auto& sec : reader.sections ) { // For all sections 948 if ( SHT_NOTE == sec->get_type() ) { // Look at notes 949 note_section_accessor notes( reader, sec.get() ); 950 Elf_Word no_notes = notes.get_notes_num(); 951 952 if ( no_notes == 0 ) 953 continue; 954 955 out << "Note section (" << sec->get_name() << ")" << std::endl 956 << " No Name Data size Description" 957 << std::endl; 958 for ( Elf_Word j = 0; j < no_notes; ++j ) { // For all notes 959 Elf_Word type; 960 std::string name; 961 char* desc; 962 Elf_Word descsz; 963 964 if ( notes.get_note( j, type, name, desc, descsz ) ) { 965 // 'name' usually contains \0 at the end. Remove it 966 name = name.c_str(); 967 note( out, j, type, name, desc, descsz ); 968 out << std::endl; 969 } 970 } 971 972 out << std::endl; 973 } 974 } 975 976 Elf_Half no = reader.segments.size(); 977 for ( Elf_Half i = 0; i < no; ++i ) { // For all segments 978 segment* seg = reader.segments[i]; 979 if ( PT_NOTE == seg->get_type() ) { // Look at notes 980 note_segment_accessor notes( reader, seg ); 981 Elf_Word no_notes = notes.get_notes_num(); 982 983 if ( no_notes == 0 ) 984 continue; 985 986 out << "Note segment (" << i << ")" << std::endl 987 << " No Name Data size Description" 988 << std::endl; 989 for ( Elf_Word j = 0; j < no_notes; ++j ) { // For all notes 990 Elf_Word type; 991 std::string name; 992 char* desc; 993 Elf_Word descsz; 994 995 if ( notes.get_note( j, type, name, desc, descsz ) ) { 996 // 'name' usually contains \0 at the end. Remove it 997 name = name.c_str(); 998 note( out, j, type, name, desc, descsz ); 999 out << std::endl; 1000 } 1001 } 1002 1003 out << std::endl; 1004 } 1005 } 1006 } 1007 1008 //------------------------------------------------------------------------------ note(std::ostream & out,int no,Elf_Word type,const std::string & name,void * desc,Elf_Word descsz)1009 static void note( std::ostream& out, 1010 int no, 1011 Elf_Word type, 1012 const std::string& name, 1013 void* desc, 1014 Elf_Word descsz ) 1015 { 1016 out << " [" << DUMP_DEC_FORMAT( 2 ) << no << "] "; 1017 1018 const auto name_group = std::find_if( 1019 std::begin( note_tag_table ), std::end( note_tag_table ), 1020 [&name]( const note_tag_t& entry ) { return entry.name == name; } ); 1021 1022 std::vector<note_tag_t::note_values_t>::const_iterator type_value; 1023 if ( name_group != std::end( note_tag_table ) ) { 1024 type_value = std::find_if( 1025 name_group->values.begin(), name_group->values.end(), 1026 [&type]( const note_tag_t::note_values_t& e ) { 1027 return e.type == type; 1028 } ); 1029 } 1030 1031 if ( name_group != std::end( note_tag_table ) && 1032 type_value != name_group->values.end() ) { 1033 out << DUMP_STR_FORMAT( 12 ) << name_group->name << " " 1034 << DUMP_HEX0x_FORMAT( 8 ) << descsz << " " 1035 << type_value->type_str << " (" << type_value->description 1036 << ")"; 1037 } 1038 else { 1039 out << DUMP_STR_FORMAT( 12 ) << name << " " 1040 << DUMP_HEX0x_FORMAT( 8 ) << descsz << " " 1041 << DUMP_HEX0x_FORMAT( 8 ) << type; 1042 } 1043 1044 if ( descsz != 0 ) { 1045 for ( Elf_Word i = 0; i < descsz; ++i ) { 1046 if ( i % 16 == 0 ) { 1047 out << std::endl << " "; 1048 } 1049 out << DUMP_HEX_FORMAT( 2 ) 1050 << (uint32_t)( (uint8_t*)( desc ) )[i]; 1051 } 1052 } 1053 } 1054 1055 //------------------------------------------------------------------------------ modinfo(std::ostream & out,const elfio & reader)1056 static void modinfo( std::ostream& out, const elfio& reader ) 1057 { 1058 for ( const auto& sec : reader.sections ) { // For all sections 1059 if ( ".modinfo" == sec->get_name() ) { // Look for the section 1060 out << "Section .modinfo" << std::endl; 1061 1062 const_modinfo_section_accessor modinfo( sec.get() ); 1063 for ( Elf_Word i = 0; i < modinfo.get_attribute_num(); i++ ) { 1064 std::string field; 1065 std::string value; 1066 if ( modinfo.get_attribute( i, field, value ) ) { 1067 out << " " << std::setw( 20 ) << field 1068 << std::setw( 0 ) << " = " << value << std::endl; 1069 } 1070 } 1071 1072 out << std::endl; 1073 break; 1074 } 1075 } 1076 } 1077 1078 //------------------------------------------------------------------------------ dynamic_tags(std::ostream & out,const elfio & reader)1079 static void dynamic_tags( std::ostream& out, const elfio& reader ) 1080 { 1081 for ( const auto& sec : reader.sections ) { // For all sections 1082 if ( SHT_DYNAMIC == sec->get_type() ) { 1083 dynamic_section_accessor dynamic( reader, sec.get() ); 1084 1085 Elf_Xword dyn_no = dynamic.get_entries_num(); 1086 if ( dyn_no == 0 ) 1087 continue; 1088 1089 out << "Dynamic section (" << sec->get_name() << ")" 1090 << std::endl; 1091 out << "[ Nr ] Tag Name/Value" << std::endl; 1092 for ( Elf_Xword i = 0; i < dyn_no; ++i ) { 1093 Elf_Xword tag = 0; 1094 Elf_Xword value = 0; 1095 std::string str; 1096 dynamic.get_entry( i, tag, value, str ); 1097 dynamic_tag( out, i, tag, value, str, reader.get_class() ); 1098 if ( DT_NULL == tag ) { 1099 break; 1100 } 1101 } 1102 1103 out << std::endl; 1104 } 1105 } 1106 } 1107 1108 //------------------------------------------------------------------------------ dynamic_tag(std::ostream & out,Elf_Xword no,Elf_Xword tag,Elf_Xword value,const std::string & str,unsigned int)1109 static void dynamic_tag( std::ostream& out, 1110 Elf_Xword no, 1111 Elf_Xword tag, 1112 Elf_Xword value, 1113 const std::string& str, 1114 unsigned int /*elf_class*/ ) 1115 { 1116 out << "[" << DUMP_DEC_FORMAT( 5 ) << no << "] " 1117 << DUMP_STR_FORMAT( 16 ) << str_dynamic_tag( tag ) << " "; 1118 if ( str.empty() ) { 1119 out << DUMP_HEX0x_FORMAT( 16 ) << value << " "; 1120 } 1121 else { 1122 out << DUMP_STR_FORMAT( 32 ) << str << " "; 1123 } 1124 out << std::endl; 1125 } 1126 1127 //------------------------------------------------------------------------------ section_data(std::ostream & out,const section * sec)1128 static void section_data( std::ostream& out, const section* sec ) 1129 { 1130 std::ios_base::fmtflags original_flags = out.flags(); 1131 1132 out << sec->get_name() << std::endl; 1133 const char* pdata = sec->get_data(); 1134 if ( pdata ) { 1135 ELFIO::Elf_Xword i; 1136 for ( i = 0; i < std::min( sec->get_size(), MAX_DATA_ENTRIES ); 1137 ++i ) { 1138 if ( i % 16 == 0 ) { 1139 out << "[" << DUMP_HEX0x_FORMAT( 8 ) << i << "]"; 1140 } 1141 1142 out << " " << DUMP_HEX0x_FORMAT( 2 ) 1143 << ( pdata[i] & 0x000000FF ); 1144 1145 if ( i % 16 == 15 ) { 1146 out << std::endl; 1147 } 1148 } 1149 if ( i % 16 != 0 ) { 1150 out << std::endl; 1151 } 1152 1153 out.flags( original_flags ); 1154 } 1155 1156 return; 1157 } 1158 1159 //------------------------------------------------------------------------------ section_datas(std::ostream & out,const elfio & reader)1160 static void section_datas( std::ostream& out, const elfio& reader ) 1161 { 1162 Elf_Half n = reader.sections.size(); 1163 1164 if ( n == 0 ) { 1165 return; 1166 } 1167 1168 out << "Section Data:" << std::endl; 1169 1170 for ( Elf_Half i = 1; i < n; ++i ) { // For all sections 1171 const section* sec = reader.sections[i]; 1172 if ( sec->get_type() == SHT_NOBITS ) { 1173 continue; 1174 } 1175 section_data( out, sec ); 1176 } 1177 1178 out << std::endl; 1179 } 1180 1181 //------------------------------------------------------------------------------ 1182 static void segment_data(std::ostream & out,Elf_Half no,const segment * seg)1183 segment_data( std::ostream& out, Elf_Half no, const segment* seg ) 1184 { 1185 std::ios_base::fmtflags original_flags = out.flags(); 1186 1187 out << "Segment # " << no << std::endl; 1188 const char* pdata = seg->get_data(); 1189 if ( pdata ) { 1190 ELFIO::Elf_Xword i; 1191 for ( i = 0; i < std::min( seg->get_file_size(), MAX_DATA_ENTRIES ); 1192 ++i ) { 1193 if ( i % 16 == 0 ) { 1194 out << "[" << DUMP_HEX0x_FORMAT( 8 ) << i << "]"; 1195 } 1196 1197 out << " " << DUMP_HEX0x_FORMAT( 2 ) 1198 << ( pdata[i] & 0x000000FF ); 1199 1200 if ( i % 16 == 15 ) { 1201 out << std::endl; 1202 } 1203 } 1204 if ( i % 16 != 0 ) { 1205 out << std::endl; 1206 } 1207 1208 out.flags( original_flags ); 1209 } 1210 1211 return; 1212 } 1213 1214 //------------------------------------------------------------------------------ segment_datas(std::ostream & out,const elfio & reader)1215 static void segment_datas( std::ostream& out, const elfio& reader ) 1216 { 1217 Elf_Half n = reader.segments.size(); 1218 1219 if ( n == 0 ) { 1220 return; 1221 } 1222 1223 out << "Segment Data:" << std::endl; 1224 1225 for ( Elf_Half i = 0; i < n; ++i ) { // For all sections 1226 const segment* seg = reader.segments[i]; 1227 segment_data( out, i, seg ); 1228 } 1229 1230 out << std::endl; 1231 } 1232 1233 //------------------------------------------------------------------------------ 1234 #define STR_FUNC_TABLE( name ) \ 1235 template <typename T> static std::string str_##name( const T key ) \ 1236 { \ 1237 return format_assoc( name##_table, key ); \ 1238 } 1239 1240 STR_FUNC_TABLE( class ) 1241 STR_FUNC_TABLE( endian ) 1242 STR_FUNC_TABLE( version ) 1243 STR_FUNC_TABLE( os_abi ) 1244 STR_FUNC_TABLE( type ) 1245 STR_FUNC_TABLE( machine ) 1246 STR_FUNC_TABLE( section_type ) 1247 STR_FUNC_TABLE( segment_type ) 1248 STR_FUNC_TABLE( segment_flag ) 1249 STR_FUNC_TABLE( symbol_bind ) 1250 STR_FUNC_TABLE( symbol_type ) 1251 STR_FUNC_TABLE( dynamic_tag ) 1252 1253 #undef STR_FUNC_TABLE 1254 1255 private: 1256 //------------------------------------------------------------------------------ 1257 template <typename T, typename K> find_value_in_table(const T & table,const K & key)1258 std::string static find_value_in_table( const T& table, const K& key ) 1259 { 1260 std::string res = "?"; 1261 for ( unsigned int i = 0; i < sizeof( table ) / sizeof( table[0] ); 1262 ++i ) { 1263 if ( table[i].key == key ) { 1264 res = table[i].str; 1265 break; 1266 } 1267 } 1268 1269 return res; 1270 } 1271 1272 //------------------------------------------------------------------------------ 1273 template <typename T, typename K> format_assoc(const T & table,const K & key)1274 static std::string format_assoc( const T& table, const K& key ) 1275 { 1276 std::string str = find_value_in_table( table, key ); 1277 if ( str == "?" ) { 1278 std::ostringstream oss; 1279 oss << str << " (0x" << std::hex << key << ")"; 1280 str = oss.str(); 1281 } 1282 1283 return str; 1284 } 1285 1286 //------------------------------------------------------------------------------ 1287 template <typename T> format_assoc(const T & table,const char key)1288 static std::string format_assoc( const T& table, const char key ) 1289 { 1290 return format_assoc( table, (const int)key ); 1291 } 1292 1293 //------------------------------------------------------------------------------ section_flags(Elf_Xword flags)1294 static std::string section_flags( Elf_Xword flags ) 1295 { 1296 std::string ret = ""; 1297 if ( flags & SHF_WRITE ) { 1298 ret += "W"; 1299 } 1300 if ( flags & SHF_ALLOC ) { 1301 ret += "A"; 1302 } 1303 if ( flags & SHF_EXECINSTR ) { 1304 ret += "X"; 1305 } 1306 if ( flags & SHF_MERGE ) { 1307 ret += "M"; 1308 } 1309 if ( flags & SHF_STRINGS ) { 1310 ret += "S"; 1311 } 1312 if ( flags & SHF_INFO_LINK ) { 1313 ret += "I"; 1314 } 1315 if ( flags & SHF_LINK_ORDER ) { 1316 ret += "L"; 1317 } 1318 if ( flags & SHF_OS_NONCONFORMING ) { 1319 ret += "O"; 1320 } 1321 if ( flags & SHF_GROUP ) { 1322 ret += "G"; 1323 } 1324 if ( flags & SHF_TLS ) { 1325 ret += "T"; 1326 } 1327 if ( flags & SHF_COMPRESSED ) { 1328 ret += "C"; 1329 } 1330 if ( flags & SHF_EXCLUDE ) { 1331 ret += "E"; 1332 } 1333 if ( flags & SHF_GNU_MBIND ) { 1334 ret += "D"; 1335 } 1336 1337 return ret; 1338 } 1339 1340 #undef DUMP_DEC_FORMAT 1341 #undef DUMP_HEX0x_FORMAT 1342 #undef DUMP_STR_FORMAT 1343 }; // class dump 1344 } // namespace ELFIO 1345 1346 #endif // ELFIO_DUMP_HPP 1347