• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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