readelf.cpp 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737
  1. /*
  2. * Copyright (c) 2020-2022, the SerenityOS developers.
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/String.h>
  7. #include <AK/StringBuilder.h>
  8. #include <AK/StringView.h>
  9. #include <LibCore/ArgsParser.h>
  10. #include <LibCore/File.h>
  11. #include <LibCore/MappedFile.h>
  12. #include <LibELF/DynamicLoader.h>
  13. #include <LibELF/DynamicObject.h>
  14. #include <LibELF/Image.h>
  15. #include <LibELF/Validation.h>
  16. #include <ctype.h>
  17. #include <fcntl.h>
  18. #include <stdio.h>
  19. #include <unistd.h>
  20. static const char* object_program_header_type_to_string(ElfW(Word) type)
  21. {
  22. switch (type) {
  23. case PT_NULL:
  24. return "NULL";
  25. case PT_LOAD:
  26. return "LOAD";
  27. case PT_DYNAMIC:
  28. return "DYNAMIC";
  29. case PT_INTERP:
  30. return "INTERP";
  31. case PT_NOTE:
  32. return "NOTE";
  33. case PT_SHLIB:
  34. return "SHLIB";
  35. case PT_PHDR:
  36. return "PHDR";
  37. case PT_TLS:
  38. return "TLS";
  39. case PT_LOOS:
  40. return "LOOS";
  41. case PT_HIOS:
  42. return "HIOS";
  43. case PT_LOPROC:
  44. return "LOPROC";
  45. case PT_HIPROC:
  46. return "HIPROC";
  47. case PT_GNU_EH_FRAME:
  48. return "GNU_EH_FRAME";
  49. case PT_GNU_RELRO:
  50. return "GNU_RELRO";
  51. case PT_GNU_STACK:
  52. return "GNU_STACK";
  53. case PT_OPENBSD_RANDOMIZE:
  54. return "OPENBSD_RANDOMIZE";
  55. case PT_OPENBSD_WXNEEDED:
  56. return "OPENBSD_WXNEEDED";
  57. case PT_OPENBSD_BOOTDATA:
  58. return "OPENBSD_BOOTDATA";
  59. default:
  60. return "(?)";
  61. }
  62. }
  63. static const char* object_section_header_type_to_string(ElfW(Word) type)
  64. {
  65. switch (type) {
  66. case SHT_NULL:
  67. return "NULL";
  68. case SHT_PROGBITS:
  69. return "PROGBITS";
  70. case SHT_SYMTAB:
  71. return "SYMTAB";
  72. case SHT_STRTAB:
  73. return "STRTAB";
  74. case SHT_RELA:
  75. return "RELA";
  76. case SHT_HASH:
  77. return "HASH";
  78. case SHT_DYNAMIC:
  79. return "DYNAMIC";
  80. case SHT_NOTE:
  81. return "NOTE";
  82. case SHT_NOBITS:
  83. return "NOBITS";
  84. case SHT_REL:
  85. return "REL";
  86. case SHT_SHLIB:
  87. return "SHLIB";
  88. case SHT_DYNSYM:
  89. return "DYNSYM";
  90. case SHT_NUM:
  91. return "NUM";
  92. case SHT_INIT_ARRAY:
  93. return "INIT_ARRAY";
  94. case SHT_FINI_ARRAY:
  95. return "FINI_ARRAY";
  96. case SHT_PREINIT_ARRAY:
  97. return "PREINIT_ARRAY";
  98. case SHT_GROUP:
  99. return "GROUP";
  100. case SHT_SYMTAB_SHNDX:
  101. return "SYMTAB_SHNDX";
  102. case SHT_LOOS:
  103. return "SOOS";
  104. case SHT_SUNW_dof:
  105. return "SUNW_dof";
  106. case SHT_GNU_LIBLIST:
  107. return "GNU_LIBLIST";
  108. case SHT_SUNW_move:
  109. return "SUNW_move";
  110. case SHT_SUNW_syminfo:
  111. return "SUNW_syminfo";
  112. case SHT_SUNW_verdef:
  113. return "SUNW_verdef";
  114. case SHT_SUNW_verneed:
  115. return "SUNW_verneed";
  116. case SHT_SUNW_versym: // or SHT_HIOS
  117. return "SUNW_versym";
  118. case SHT_LOPROC:
  119. return "LOPROC";
  120. case SHT_HIPROC:
  121. return "HIPROC";
  122. case SHT_LOUSER:
  123. return "LOUSER";
  124. case SHT_HIUSER:
  125. return "HIUSER";
  126. case SHT_GNU_HASH:
  127. return "GNU_HASH";
  128. default:
  129. return "(?)";
  130. }
  131. }
  132. static const char* object_symbol_type_to_string(ElfW(Word) type)
  133. {
  134. switch (type) {
  135. case STT_NOTYPE:
  136. return "NOTYPE";
  137. case STT_OBJECT:
  138. return "OBJECT";
  139. case STT_FUNC:
  140. return "FUNC";
  141. case STT_SECTION:
  142. return "SECTION";
  143. case STT_FILE:
  144. return "FILE";
  145. case STT_TLS:
  146. return "TLS";
  147. case STT_LOPROC:
  148. return "LOPROC";
  149. case STT_HIPROC:
  150. return "HIPROC";
  151. default:
  152. return "(?)";
  153. }
  154. }
  155. static const char* object_symbol_binding_to_string(ElfW(Word) type)
  156. {
  157. switch (type) {
  158. case STB_LOCAL:
  159. return "LOCAL";
  160. case STB_GLOBAL:
  161. return "GLOBAL";
  162. case STB_WEAK:
  163. return "WEAK";
  164. case STB_NUM:
  165. return "NUM";
  166. case STB_LOPROC:
  167. return "LOPROC";
  168. case STB_HIPROC:
  169. return "HIPROC";
  170. default:
  171. return "(?)";
  172. }
  173. }
  174. static const char* object_relocation_type_to_string(ElfW(Word) type)
  175. {
  176. switch (type) {
  177. #if ARCH(I386)
  178. case R_386_NONE:
  179. return "R_386_NONE";
  180. case R_386_32:
  181. return "R_386_32";
  182. case R_386_PC32:
  183. return "R_386_PC32";
  184. case R_386_GOT32:
  185. return "R_386_GOT32";
  186. case R_386_PLT32:
  187. return "R_386_PLT32";
  188. case R_386_COPY:
  189. return "R_386_COPY";
  190. case R_386_GLOB_DAT:
  191. return "R_386_GLOB_DAT";
  192. case R_386_JMP_SLOT:
  193. return "R_386_JMP_SLOT";
  194. case R_386_RELATIVE:
  195. return "R_386_RELATIVE";
  196. case R_386_TLS_TPOFF:
  197. return "R_386_TLS_TPOFF";
  198. case R_386_TLS_TPOFF32:
  199. return "R_386_TLS_TPOFF32";
  200. #else
  201. case R_X86_64_NONE:
  202. return "R_X86_64_NONE";
  203. case R_X86_64_64:
  204. return "R_X86_64";
  205. case R_X86_64_GLOB_DAT:
  206. return "R_x86_64_GLOB_DAT";
  207. case R_X86_64_JUMP_SLOT:
  208. return "R_X86_64_JUMP_SLOT";
  209. case R_X86_64_RELATIVE:
  210. return "R_X86_64_RELATIVE";
  211. case R_X86_64_TPOFF64:
  212. return "R_X86_64_TPOFF64";
  213. #endif
  214. default:
  215. return "(?)";
  216. }
  217. }
  218. int main(int argc, char** argv)
  219. {
  220. if (pledge("stdio rpath", nullptr) < 0) {
  221. perror("pledge");
  222. return 1;
  223. }
  224. const char* path;
  225. static bool display_all = false;
  226. static bool display_elf_header = false;
  227. static bool display_program_headers = false;
  228. static bool display_section_headers = false;
  229. static bool display_headers = false;
  230. static bool display_symbol_table = false;
  231. static bool display_dynamic_symbol_table = false;
  232. static bool display_core_notes = false;
  233. static bool display_relocations = false;
  234. static bool display_unwind_info = false;
  235. static bool display_dynamic_section = false;
  236. static bool display_hardening = false;
  237. StringView string_dump_section {};
  238. Core::ArgsParser args_parser;
  239. args_parser.add_option(display_all, "Display all", "all", 'a');
  240. args_parser.add_option(display_elf_header, "Display ELF header", "file-header", 'h');
  241. args_parser.add_option(display_program_headers, "Display program headers", "program-headers", 'l');
  242. args_parser.add_option(display_section_headers, "Display section headers", "section-headers", 'S');
  243. args_parser.add_option(display_headers, "Equivalent to: -h -l -S -s -r -d -n -u -c", "headers", 'e');
  244. args_parser.add_option(display_symbol_table, "Display the symbol table", "syms", 's');
  245. args_parser.add_option(display_dynamic_symbol_table, "Display the dynamic symbol table", "dyn-syms", '\0');
  246. args_parser.add_option(display_dynamic_section, "Display the dynamic section", "dynamic", 'd');
  247. args_parser.add_option(display_core_notes, "Display core notes", "notes", 'n');
  248. args_parser.add_option(display_relocations, "Display relocations", "relocs", 'r');
  249. args_parser.add_option(display_unwind_info, "Display unwind info", "unwind", 'u');
  250. args_parser.add_option(display_hardening, "Display security hardening info", "checksec", 'c');
  251. args_parser.add_option(string_dump_section, "Display the contents of a section as strings", "string-dump", 'p', "section-name");
  252. args_parser.add_positional_argument(path, "ELF path", "path");
  253. args_parser.parse(argc, argv);
  254. if (argc < 3) {
  255. args_parser.print_usage(stderr, argv[0]);
  256. return -1;
  257. }
  258. if (display_headers) {
  259. display_elf_header = true;
  260. display_program_headers = true;
  261. display_section_headers = true;
  262. }
  263. if (display_all) {
  264. display_elf_header = true;
  265. display_program_headers = true;
  266. display_section_headers = true;
  267. display_dynamic_symbol_table = true;
  268. display_dynamic_section = true;
  269. display_core_notes = true;
  270. display_relocations = true;
  271. display_unwind_info = true;
  272. display_symbol_table = true;
  273. display_hardening = true;
  274. }
  275. auto file_or_error = Core::MappedFile::map(path);
  276. if (file_or_error.is_error()) {
  277. warnln("Unable to map file {}: {}", path, file_or_error.error());
  278. return -1;
  279. }
  280. auto elf_image_data = file_or_error.value()->bytes();
  281. ELF::Image elf_image(elf_image_data);
  282. if (!elf_image.is_valid()) {
  283. warnln("File is not a valid ELF object");
  284. return -1;
  285. }
  286. StringBuilder interpreter_path_builder;
  287. auto result_or_error = ELF::validate_program_headers(*(const ElfW(Ehdr)*)elf_image_data.data(), elf_image_data.size(), elf_image_data, &interpreter_path_builder);
  288. if (result_or_error.is_error() || !result_or_error.value()) {
  289. warnln("Invalid ELF headers");
  290. return -1;
  291. }
  292. auto interpreter_path = interpreter_path_builder.string_view();
  293. auto& header = *reinterpret_cast<const ElfW(Ehdr)*>(elf_image_data.data());
  294. RefPtr<ELF::DynamicObject> object = nullptr;
  295. if (elf_image.is_dynamic()) {
  296. if (interpreter_path.is_empty()) {
  297. interpreter_path = "/usr/lib/Loader.so"sv;
  298. warnln("Warning: Dynamic ELF object has no interpreter path. Using: {}", interpreter_path);
  299. }
  300. auto interpreter_file_or_error = Core::MappedFile::map(interpreter_path);
  301. if (interpreter_file_or_error.is_error()) {
  302. warnln("Unable to map interpreter file {}: {}", interpreter_path, interpreter_file_or_error.error());
  303. return -1;
  304. }
  305. auto interpreter_image_data = interpreter_file_or_error.value()->bytes();
  306. ELF::Image interpreter_image(interpreter_image_data);
  307. if (!interpreter_image.is_valid()) {
  308. warnln("ELF interpreter image is invalid");
  309. return -1;
  310. }
  311. int fd = open(path, O_RDONLY);
  312. if (fd < 0) {
  313. outln("Unable to open file {}", path);
  314. return 1;
  315. }
  316. auto result = ELF::DynamicLoader::try_create(fd, path);
  317. if (result.is_error()) {
  318. outln("{}", result.error().text);
  319. return 1;
  320. }
  321. auto& loader = result.value();
  322. if (!loader->is_valid()) {
  323. outln("{} is not a valid ELF dynamic shared object!", path);
  324. return 1;
  325. }
  326. object = loader->map();
  327. if (!object) {
  328. outln("Failed to map dynamic ELF object {}", path);
  329. return 1;
  330. }
  331. }
  332. if (display_elf_header) {
  333. outln("ELF header:");
  334. out(" Magic: ");
  335. for (char i : StringView { header.e_ident, sizeof(header.e_ident) }) {
  336. if (isprint(i)) {
  337. out("{:c} ", i);
  338. } else {
  339. out("{:02x} ", i);
  340. }
  341. }
  342. outln();
  343. outln(" Type: {} ({})", header.e_type, ELF::Image::object_file_type_to_string(header.e_type).value_or("(?)"));
  344. outln(" Machine: {} ({})", header.e_machine, ELF::Image::object_machine_type_to_string(header.e_machine).value_or("(?)"));
  345. outln(" Version: {:#x}", header.e_version);
  346. outln(" Entry point address: {:#x}", header.e_entry);
  347. outln(" Start of program headers: {} (bytes into file)", header.e_phoff);
  348. outln(" Start of section headers: {} (bytes into file)", header.e_shoff);
  349. outln(" Flags: {:#x}", header.e_flags);
  350. outln(" Size of this header: {} (bytes)", header.e_ehsize);
  351. outln(" Size of program headers: {} (bytes)", header.e_phentsize);
  352. outln(" Number of program headers: {}", header.e_phnum);
  353. outln(" Size of section headers: {} (bytes)", header.e_shentsize);
  354. outln(" Number of section headers: {}", header.e_shnum);
  355. outln(" Section header string table index: {}", header.e_shstrndx);
  356. outln();
  357. }
  358. #if ARCH(I386)
  359. auto addr_padding = "";
  360. #else
  361. auto addr_padding = " ";
  362. #endif
  363. if (display_section_headers) {
  364. if (!display_all) {
  365. outln("There are {} section headers, starting at offset {:#x}:", header.e_shnum, header.e_shoff);
  366. outln();
  367. }
  368. if (!elf_image.section_count()) {
  369. outln("There are no sections in this file.");
  370. } else {
  371. outln("Section Headers:");
  372. outln(" Name Type Address{} Offset{} Size{} Flags", addr_padding, addr_padding, addr_padding);
  373. elf_image.for_each_section([](const ELF::Image::Section& section) {
  374. out(" {:19} ", section.name());
  375. out("{:15} ", object_section_header_type_to_string(section.type()));
  376. out("{:p} ", section.address());
  377. out("{:p} ", section.offset());
  378. out("{:p} ", section.size());
  379. out("{}", section.flags());
  380. outln();
  381. });
  382. }
  383. outln();
  384. }
  385. if (display_program_headers) {
  386. if (!display_all) {
  387. outln("ELF file type is {} ({})", header.e_type, ELF::Image::object_file_type_to_string(header.e_type).value_or("(?)"));
  388. outln("Entry point {:#x}\n", header.e_entry);
  389. outln("There are {} program headers, starting at offset {}", header.e_phnum, header.e_phoff);
  390. outln();
  391. }
  392. if (!elf_image.program_header_count()) {
  393. outln("There are no program headers in this file.");
  394. } else {
  395. outln("Program Headers:");
  396. outln(" Type Offset{} VirtAddr{} PhysAddr{} FileSiz{} MemSiz{} Flg Align",
  397. addr_padding, addr_padding, addr_padding, addr_padding, addr_padding);
  398. elf_image.for_each_program_header([](const ELF::Image::ProgramHeader& program_header) {
  399. out(" ");
  400. out("{:14} ", object_program_header_type_to_string(program_header.type()));
  401. out("{:p} ", program_header.offset());
  402. out("{:p} ", program_header.vaddr().as_ptr());
  403. out("{:p} ", program_header.vaddr().as_ptr()); // FIXME: assumes PhysAddr = VirtAddr
  404. out("{:p} ", program_header.size_in_image());
  405. out("{:p} ", program_header.size_in_memory());
  406. out("{:04x} ", program_header.flags());
  407. out("{:p}", program_header.alignment());
  408. outln();
  409. if (program_header.type() == PT_INTERP)
  410. outln(" [Interpreter: {}]", program_header.raw_data());
  411. });
  412. }
  413. // TODO: Display section to segment mapping
  414. outln();
  415. }
  416. if (display_dynamic_section) {
  417. auto found_dynamic_section = false;
  418. if (elf_image.is_dynamic()) {
  419. elf_image.for_each_section([&found_dynamic_section](const ELF::Image::Section& section) {
  420. if (section.name() != ELF_DYNAMIC)
  421. return IterationDecision::Continue;
  422. found_dynamic_section = true;
  423. if (section.entry_count()) {
  424. outln("Dynamic section '{}' at offset {:#08x} contains {} entries.", section.name().to_string(), section.offset(), section.entry_count());
  425. } else {
  426. outln("Dynamic section '{}' at offset {:#08x} contains zero entries.", section.name().to_string(), section.offset());
  427. }
  428. return IterationDecision::Break;
  429. });
  430. Vector<String> libraries;
  431. object->for_each_needed_library([&libraries](StringView entry) {
  432. libraries.append(String::formatted("{}", entry));
  433. });
  434. auto library_index = 0;
  435. outln(" Tag Type Name / Value");
  436. object->for_each_dynamic_entry([&library_index, &libraries, &object](const ELF::DynamicObject::DynamicEntry& entry) {
  437. out(" {:#08x} ", entry.tag());
  438. out("{:17} ", ELF::DynamicObject::name_for_dtag(entry.tag()));
  439. if (entry.tag() == DT_NEEDED) {
  440. outln("Shared library: {}", libraries[library_index]);
  441. library_index++;
  442. } else if (entry.tag() == DT_RPATH) {
  443. outln("Library rpath: {}", object->rpath());
  444. } else if (entry.tag() == DT_RUNPATH) {
  445. outln("Library runpath: {}", object->runpath());
  446. } else if (entry.tag() == DT_SONAME) {
  447. outln("Library soname: {}", object->soname());
  448. } else {
  449. outln("{:#08x}", entry.val());
  450. }
  451. });
  452. }
  453. if (!found_dynamic_section)
  454. outln("No dynamic section in this file.");
  455. outln();
  456. }
  457. if (display_relocations) {
  458. if (elf_image.is_dynamic()) {
  459. if (!object->relocation_section().entry_count()) {
  460. outln("Relocation section '{}' at offset {:#08x} contains zero entries:", object->relocation_section().name(), object->relocation_section().offset());
  461. } else {
  462. outln("Relocation section '{}' at offset {:#08x} contains {} entries:", object->relocation_section().name(), object->relocation_section().offset(), object->relocation_section().entry_count());
  463. outln(" Offset{} Type Sym Value{} Sym Name", addr_padding, addr_padding);
  464. object->relocation_section().for_each_relocation([](const ELF::DynamicObject::Relocation& reloc) {
  465. out(" {:p} ", reloc.offset());
  466. out(" {:18} ", object_relocation_type_to_string(reloc.type()));
  467. out(" {:p} ", reloc.symbol().value());
  468. out(" {}", reloc.symbol().name());
  469. outln();
  470. });
  471. }
  472. outln();
  473. if (!object->plt_relocation_section().entry_count()) {
  474. outln("Relocation section '{}' at offset {:#08x} contains zero entries:", object->plt_relocation_section().name(), object->plt_relocation_section().offset());
  475. } else {
  476. outln("Relocation section '{}' at offset {:#08x} contains {} entries:", object->plt_relocation_section().name(), object->plt_relocation_section().offset(), object->plt_relocation_section().entry_count());
  477. outln(" Offset{} Type Sym Value{} Sym Name", addr_padding, addr_padding);
  478. object->plt_relocation_section().for_each_relocation([](const ELF::DynamicObject::Relocation& reloc) {
  479. out(" {:p} ", reloc.offset());
  480. out(" {:18} ", object_relocation_type_to_string(reloc.type()));
  481. out(" {:p} ", reloc.symbol().value());
  482. out(" {}", reloc.symbol().name());
  483. outln();
  484. });
  485. }
  486. } else {
  487. outln("No relocations in this file.");
  488. }
  489. outln();
  490. }
  491. if (display_unwind_info) {
  492. // TODO: Unwind info
  493. outln("Decoding of unwind sections for machine type {} is not supported.", ELF::Image::object_machine_type_to_string(header.e_machine).value_or("?"));
  494. outln();
  495. }
  496. if (display_core_notes) {
  497. auto found_notes = false;
  498. elf_image.for_each_program_header([&found_notes](const ELF::Image::ProgramHeader& program_header) {
  499. if (program_header.type() != PT_NOTE)
  500. return;
  501. found_notes = true;
  502. outln("Displaying notes section '{}' at offset {:#08x} of length {:#08x}:", object_program_header_type_to_string(program_header.type()), program_header.offset(), program_header.size_in_image());
  503. // FIXME: Parse CORE notes. Notes are in JSON format on SerenityOS, but vary between systems.
  504. outln("{}", program_header.raw_data());
  505. });
  506. if (!found_notes)
  507. outln("No core notes in this file.");
  508. outln();
  509. }
  510. if (display_dynamic_symbol_table || display_symbol_table) {
  511. auto found_dynamic_symbol_table = false;
  512. if (elf_image.is_dynamic()) {
  513. elf_image.for_each_section([&found_dynamic_symbol_table](const ELF::Image::Section& section) {
  514. if (section.name() != ELF_DYNSYM)
  515. return IterationDecision::Continue;
  516. found_dynamic_symbol_table = true;
  517. if (!section.entry_count()) {
  518. outln("Symbol table '{}' contains zero entries.", ELF_DYNSYM);
  519. } else {
  520. outln("Symbol table '{}' contains {} entries.", ELF_DYNSYM, section.entry_count());
  521. }
  522. return IterationDecision::Break;
  523. });
  524. if (object->symbol_count()) {
  525. // FIXME: Add support for init/fini/start/main sections
  526. outln(" Num: Value{} Size{} Type Bind Name", addr_padding, addr_padding);
  527. object->for_each_symbol([](const ELF::DynamicObject::Symbol& sym) {
  528. out(" {:>4}: ", sym.index());
  529. out("{:p} ", sym.value());
  530. out("{:p} ", sym.size());
  531. out("{:8} ", object_symbol_type_to_string(sym.type()));
  532. out("{:8} ", object_symbol_binding_to_string(sym.bind()));
  533. out("{}", sym.name());
  534. outln();
  535. });
  536. }
  537. }
  538. if (!found_dynamic_symbol_table)
  539. outln("No dynamic symbol information for this file.");
  540. outln();
  541. }
  542. if (display_symbol_table) {
  543. if (elf_image.symbol_count()) {
  544. outln("Symbol table '{}' contains {} entries:", ELF_SYMTAB, elf_image.symbol_count());
  545. outln(" Num: Value{} Size{} Type Bind Name", addr_padding, addr_padding);
  546. elf_image.for_each_symbol([](const ELF::Image::Symbol& sym) {
  547. out(" {:>4}: ", sym.index());
  548. out("{:p} ", sym.value());
  549. out("{:p} ", sym.size());
  550. out("{:8} ", object_symbol_type_to_string(sym.type()));
  551. out("{:8} ", object_symbol_binding_to_string(sym.bind()));
  552. out("{}", sym.name());
  553. outln();
  554. });
  555. } else {
  556. outln("Symbol table '{}' contains zero entries.", ELF_SYMTAB);
  557. }
  558. outln();
  559. }
  560. if (display_hardening) {
  561. outln("Security Hardening:");
  562. outln("RELRO Stack Canary NX PIE RPATH RUNPATH Symbols ");
  563. bool relro = false;
  564. elf_image.for_each_program_header([&relro](const ELF::Image::ProgramHeader& program_header) {
  565. if (program_header.type() == PT_GNU_RELRO) {
  566. relro = true;
  567. return IterationDecision::Break;
  568. }
  569. return IterationDecision::Continue;
  570. });
  571. bool full_relro = false;
  572. if (relro) {
  573. object->for_each_dynamic_entry([&full_relro](const ELF::DynamicObject::DynamicEntry& entry) {
  574. if (entry.tag() == DT_BIND_NOW) {
  575. full_relro = true;
  576. return IterationDecision::Break;
  577. }
  578. return IterationDecision::Continue;
  579. });
  580. if (full_relro)
  581. out("\033[0;32m{:13}\033[0m ", "Full RELRO");
  582. else
  583. out("\033[0;33m{:13}\033[0m ", "Partial RELRO");
  584. } else {
  585. out("\033[0;31m{:13}\033[0m ", "No RELRO");
  586. }
  587. bool canary = false;
  588. elf_image.for_each_symbol([&canary](const ELF::Image::Symbol& sym) {
  589. if (sym.name() == "__stack_chk_fail" || sym.name() == "__intel_security_cookie") {
  590. canary = true;
  591. return IterationDecision::Break;
  592. }
  593. return IterationDecision::Continue;
  594. });
  595. if (canary)
  596. out("\033[0;32m{:12}\033[0m ", "Canary found");
  597. else
  598. out("\033[0;31m{:12}\033[0m ", "No canary");
  599. bool nx = false;
  600. elf_image.for_each_program_header([&nx](const ELF::Image::ProgramHeader& program_header) {
  601. if (program_header.type() == PT_GNU_STACK) {
  602. if (program_header.flags() & PF_X)
  603. nx = false;
  604. else
  605. nx = true;
  606. return IterationDecision::Break;
  607. }
  608. return IterationDecision::Continue;
  609. });
  610. if (nx)
  611. out("\033[0;32m{:12}\033[0m ", "NX enabled");
  612. else
  613. out("\033[0;31m{:12}\033[0m ", "NX disabled");
  614. bool pie = false;
  615. if (header.e_type == ET_REL || header.e_type == ET_DYN)
  616. pie = true;
  617. if (pie)
  618. out("\033[0;32m{:12}\033[0m ", "PIE enabled");
  619. else
  620. out("\033[0;31m{:12}\033[0m ", "No PIE");
  621. StringView rpath;
  622. if (elf_image.is_dynamic())
  623. rpath = object->rpath();
  624. if (rpath.is_empty())
  625. out("\033[0;32m{:12}\033[0m ", "No RPATH");
  626. else
  627. out("\033[0;31m{:12}\033[0m ", rpath);
  628. StringView runpath;
  629. if (elf_image.is_dynamic())
  630. runpath = object->runpath();
  631. if (runpath.is_empty())
  632. out("\033[0;32m{:12}\033[0m ", "No RUNPATH");
  633. else
  634. out("\033[0;31m{:12}\033[0m ", runpath);
  635. out("{} symbols", elf_image.symbol_count());
  636. outln();
  637. }
  638. if (!string_dump_section.is_null()) {
  639. auto maybe_section = elf_image.lookup_section(string_dump_section);
  640. if (maybe_section.has_value()) {
  641. outln("String dump of section \'{}\':", string_dump_section);
  642. StringView data(maybe_section->raw_data(), maybe_section->size());
  643. data.for_each_split_view('\0', false, [&data](auto string) {
  644. auto offset = string.characters_without_null_termination() - data.characters_without_null_termination();
  645. outln("[{:6x}] {}", offset, string);
  646. });
  647. } else {
  648. warnln("Could not find section \'{}\'", string_dump_section);
  649. return 1;
  650. }
  651. }
  652. return 0;
  653. }