mirror of
https://github.com/SerenityOS/serenity.git
synced 2025-01-24 18:32:28 -05:00
f40ee1b03f
This implements more of the dlfcn functionality. Most notably: * It's now possible to dlopen() libraries which were already loaded at program startup time. This does not cause those libraries to be loaded twice. * Errors are reported via dlerror() rather than by crashing the program. * Calls to the dl*() functions are thread-safe.
842 lines
30 KiB
C++
842 lines
30 KiB
C++
/*
|
|
* Copyright (c) 2020, the SerenityOS developers.
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <AK/MappedFile.h>
|
|
#include <AK/String.h>
|
|
#include <AK/StringBuilder.h>
|
|
#include <AK/StringView.h>
|
|
#include <LibCore/ArgsParser.h>
|
|
#include <LibCore/File.h>
|
|
#include <LibELF/DynamicLoader.h>
|
|
#include <LibELF/DynamicObject.h>
|
|
#include <LibELF/Image.h>
|
|
#include <LibELF/Validation.h>
|
|
#include <ctype.h>
|
|
#include <fcntl.h>
|
|
#include <stdio.h>
|
|
#include <unistd.h>
|
|
|
|
static const char* object_file_type_to_string(Elf32_Half type)
|
|
{
|
|
switch (type) {
|
|
case ET_NONE:
|
|
return "None";
|
|
case ET_REL:
|
|
return "Relocatable";
|
|
case ET_EXEC:
|
|
return "Executable";
|
|
case ET_DYN:
|
|
return "Shared object";
|
|
case ET_CORE:
|
|
return "Core";
|
|
default:
|
|
return "(?)";
|
|
}
|
|
}
|
|
|
|
static const char* object_machine_type_to_string(Elf32_Half type)
|
|
{
|
|
switch (type) {
|
|
case ET_NONE:
|
|
return "None";
|
|
case EM_M32:
|
|
return "AT&T WE 32100";
|
|
case EM_SPARC:
|
|
return "SPARC";
|
|
case EM_386:
|
|
return "Intel 80386";
|
|
case EM_68K:
|
|
return "Motorola 68000";
|
|
case EM_88K:
|
|
return "Motorola 88000";
|
|
case EM_486:
|
|
return "Intel 80486";
|
|
case EM_860:
|
|
return "Intel 80860";
|
|
case EM_MIPS:
|
|
return "MIPS R3000 Big-Endian only";
|
|
default:
|
|
return "(?)";
|
|
}
|
|
}
|
|
|
|
static const char* object_program_header_type_to_string(Elf32_Word type)
|
|
{
|
|
switch (type) {
|
|
case PT_NULL:
|
|
return "NULL";
|
|
case PT_LOAD:
|
|
return "LOAD";
|
|
case PT_DYNAMIC:
|
|
return "DYNAMIC";
|
|
case PT_INTERP:
|
|
return "INTERP";
|
|
case PT_NOTE:
|
|
return "NOTE";
|
|
case PT_SHLIB:
|
|
return "SHLIB";
|
|
case PT_PHDR:
|
|
return "PHDR";
|
|
case PT_TLS:
|
|
return "TLS";
|
|
case PT_LOOS:
|
|
return "LOOS";
|
|
case PT_HIOS:
|
|
return "HIOS";
|
|
case PT_LOPROC:
|
|
return "LOPROC";
|
|
case PT_HIPROC:
|
|
return "HIPROC";
|
|
case PT_GNU_EH_FRAME:
|
|
return "GNU_EH_FRAME";
|
|
case PT_GNU_RELRO:
|
|
return "GNU_RELRO";
|
|
case PT_GNU_STACK:
|
|
return "GNU_STACK";
|
|
case PT_OPENBSD_RANDOMIZE:
|
|
return "OPENBSD_RANDOMIZE";
|
|
case PT_OPENBSD_WXNEEDED:
|
|
return "OPENBSD_WXNEEDED";
|
|
case PT_OPENBSD_BOOTDATA:
|
|
return "OPENBSD_BOOTDATA";
|
|
default:
|
|
return "(?)";
|
|
}
|
|
}
|
|
|
|
static const char* object_section_header_type_to_string(Elf32_Word type)
|
|
{
|
|
switch (type) {
|
|
case SHT_NULL:
|
|
return "NULL";
|
|
case SHT_PROGBITS:
|
|
return "PROGBITS";
|
|
case SHT_SYMTAB:
|
|
return "SYMTAB";
|
|
case SHT_STRTAB:
|
|
return "STRTAB";
|
|
case SHT_RELA:
|
|
return "RELA";
|
|
case SHT_HASH:
|
|
return "HASH";
|
|
case SHT_DYNAMIC:
|
|
return "DYNAMIC";
|
|
case SHT_NOTE:
|
|
return "NOTE";
|
|
case SHT_NOBITS:
|
|
return "NOBITS";
|
|
case SHT_REL:
|
|
return "REL";
|
|
case SHT_SHLIB:
|
|
return "SHLIB";
|
|
case SHT_DYNSYM:
|
|
return "DYNSYM";
|
|
case SHT_NUM:
|
|
return "NUM";
|
|
case SHT_INIT_ARRAY:
|
|
return "INIT_ARRAY";
|
|
case SHT_FINI_ARRAY:
|
|
return "FINI_ARRAY";
|
|
case SHT_PREINIT_ARRAY:
|
|
return "PREINIT_ARRAY";
|
|
case SHT_GROUP:
|
|
return "GROUP";
|
|
case SHT_SYMTAB_SHNDX:
|
|
return "SYMTAB_SHNDX";
|
|
case SHT_LOOS:
|
|
return "SOOS";
|
|
case SHT_SUNW_dof:
|
|
return "SUNW_dof";
|
|
case SHT_GNU_LIBLIST:
|
|
return "GNU_LIBLIST";
|
|
case SHT_SUNW_move:
|
|
return "SUNW_move";
|
|
case SHT_SUNW_syminfo:
|
|
return "SUNW_syminfo";
|
|
case SHT_SUNW_verdef:
|
|
return "SUNW_verdef";
|
|
case SHT_SUNW_verneed:
|
|
return "SUNW_verneed";
|
|
case SHT_SUNW_versym: // or SHT_HIOS
|
|
return "SUNW_versym";
|
|
case SHT_LOPROC:
|
|
return "LOPROC";
|
|
case SHT_HIPROC:
|
|
return "HIPROC";
|
|
case SHT_LOUSER:
|
|
return "LOUSER";
|
|
case SHT_HIUSER:
|
|
return "HIUSER";
|
|
case SHT_GNU_HASH:
|
|
return "GNU_HASH";
|
|
default:
|
|
return "(?)";
|
|
}
|
|
}
|
|
|
|
static const char* object_symbol_type_to_string(Elf32_Word type)
|
|
{
|
|
switch (type) {
|
|
case STT_NOTYPE:
|
|
return "NOTYPE";
|
|
case STT_OBJECT:
|
|
return "OBJECT";
|
|
case STT_FUNC:
|
|
return "FUNC";
|
|
case STT_SECTION:
|
|
return "SECTION";
|
|
case STT_FILE:
|
|
return "FILE";
|
|
case STT_TLS:
|
|
return "TLS";
|
|
case STT_LOPROC:
|
|
return "LOPROC";
|
|
case STT_HIPROC:
|
|
return "HIPROC";
|
|
default:
|
|
return "(?)";
|
|
}
|
|
}
|
|
|
|
static const char* object_symbol_binding_to_string(Elf32_Word type)
|
|
{
|
|
switch (type) {
|
|
case STB_LOCAL:
|
|
return "LOCAL";
|
|
case STB_GLOBAL:
|
|
return "GLOBAL";
|
|
case STB_WEAK:
|
|
return "WEAK";
|
|
case STB_NUM:
|
|
return "NUM";
|
|
case STB_LOPROC:
|
|
return "LOPROC";
|
|
case STB_HIPROC:
|
|
return "HIPROC";
|
|
default:
|
|
return "(?)";
|
|
}
|
|
}
|
|
|
|
static const char* object_relocation_type_to_string(Elf32_Word type)
|
|
{
|
|
switch (type) {
|
|
case R_386_NONE:
|
|
return "R_386_NONE";
|
|
case R_386_32:
|
|
return "R_386_32";
|
|
case R_386_PC32:
|
|
return "R_386_PC32";
|
|
case R_386_GOT32:
|
|
return "R_386_GOT32";
|
|
case R_386_PLT32:
|
|
return "R_386_PLT32";
|
|
case R_386_COPY:
|
|
return "R_386_COPY";
|
|
case R_386_GLOB_DAT:
|
|
return "R_386_GLOB_DAT";
|
|
case R_386_JMP_SLOT:
|
|
return "R_386_JMP_SLOT";
|
|
case R_386_RELATIVE:
|
|
return "R_386_RELATIVE";
|
|
case R_386_TLS_TPOFF:
|
|
return "R_386_TLS_TPOFF";
|
|
case R_386_TLS_TPOFF32:
|
|
return "R_386_TLS_TPOFF32";
|
|
default:
|
|
return "(?)";
|
|
}
|
|
}
|
|
|
|
static const char* object_tag_to_string(Elf32_Sword dt_tag)
|
|
{
|
|
switch (dt_tag) {
|
|
case DT_NULL:
|
|
return "NULL"; /* marks end of _DYNAMIC array */
|
|
case DT_NEEDED:
|
|
return "NEEDED"; /* string table offset of needed lib */
|
|
case DT_PLTRELSZ:
|
|
return "PLTRELSZ"; /* size of relocation entries in PLT */
|
|
case DT_PLTGOT:
|
|
return "PLTGOT"; /* address PLT/GOT */
|
|
case DT_HASH:
|
|
return "HASH"; /* address of symbol hash table */
|
|
case DT_STRTAB:
|
|
return "STRTAB"; /* address of string table */
|
|
case DT_SYMTAB:
|
|
return "SYMTAB"; /* address of symbol table */
|
|
case DT_RELA:
|
|
return "RELA"; /* address of relocation table */
|
|
case DT_RELASZ:
|
|
return "RELASZ"; /* size of relocation table */
|
|
case DT_RELAENT:
|
|
return "RELAENT"; /* size of relocation entry */
|
|
case DT_STRSZ:
|
|
return "STRSZ"; /* size of string table */
|
|
case DT_SYMENT:
|
|
return "SYMENT"; /* size of symbol table entry */
|
|
case DT_INIT:
|
|
return "INIT"; /* address of initialization func. */
|
|
case DT_FINI:
|
|
return "FINI"; /* address of termination function */
|
|
case DT_SONAME:
|
|
return "SONAME"; /* string table offset of shared obj */
|
|
case DT_RPATH:
|
|
return "RPATH"; /* string table offset of library search path */
|
|
case DT_SYMBOLIC:
|
|
return "SYMBOLIC"; /* start sym search in shared obj. */
|
|
case DT_REL:
|
|
return "REL"; /* address of rel. tbl. w addends */
|
|
case DT_RELSZ:
|
|
return "RELSZ"; /* size of DT_REL relocation table */
|
|
case DT_RELENT:
|
|
return "RELENT"; /* size of DT_REL relocation entry */
|
|
case DT_PLTREL:
|
|
return "PLTREL"; /* PLT referenced relocation entry */
|
|
case DT_DEBUG:
|
|
return "DEBUG"; /* bugger */
|
|
case DT_TEXTREL:
|
|
return "TEXTREL"; /* Allow rel. mod. to unwritable seg */
|
|
case DT_JMPREL:
|
|
return "JMPREL"; /* add. of PLT's relocation entries */
|
|
case DT_BIND_NOW:
|
|
return "BIND_NOW"; /* Bind now regardless of env setting */
|
|
case DT_INIT_ARRAY:
|
|
return "INIT_ARRAY"; /* address of array of init func */
|
|
case DT_FINI_ARRAY:
|
|
return "FINI_ARRAY"; /* address of array of term func */
|
|
case DT_INIT_ARRAYSZ:
|
|
return "INIT_ARRAYSZ"; /* size of array of init func */
|
|
case DT_FINI_ARRAYSZ:
|
|
return "FINI_ARRAYSZ"; /* size of array of term func */
|
|
case DT_RUNPATH:
|
|
return "RUNPATH"; /* strtab offset of lib search path */
|
|
case DT_FLAGS:
|
|
return "FLAGS"; /* Set of DF_* flags */
|
|
case DT_ENCODING:
|
|
return "ENCODING"; /* further DT_* follow encoding rules */
|
|
case DT_PREINIT_ARRAY:
|
|
return "PREINIT_ARRAY"; /* address of array of preinit func */
|
|
case DT_PREINIT_ARRAYSZ:
|
|
return "PREINIT_ARRAYSZ"; /* size of array of preinit func */
|
|
case DT_LOOS:
|
|
return "LOOS"; /* reserved range for OS */
|
|
case DT_HIOS:
|
|
return "HIOS"; /* specific dynamic array tags */
|
|
case DT_LOPROC:
|
|
return "LOPROC"; /* reserved range for processor */
|
|
case DT_HIPROC:
|
|
return "HIPROC"; /* specific dynamic array tags */
|
|
case DT_GNU_HASH:
|
|
return "GNU_HASH"; /* address of GNU hash table */
|
|
case DT_RELACOUNT:
|
|
return "RELACOUNT"; /* if present, number of RELATIVE */
|
|
case DT_RELCOUNT:
|
|
return "RELCOUNT"; /* relocs, which must come first */
|
|
case DT_FLAGS_1:
|
|
return "FLAGS_1";
|
|
default:
|
|
return "??";
|
|
}
|
|
}
|
|
|
|
int main(int argc, char** argv)
|
|
{
|
|
if (pledge("stdio rpath", nullptr) < 0) {
|
|
perror("pledge");
|
|
return 1;
|
|
}
|
|
|
|
const char* path;
|
|
static bool display_all = false;
|
|
static bool display_elf_header = false;
|
|
static bool display_program_headers = false;
|
|
static bool display_section_headers = false;
|
|
static bool display_headers = false;
|
|
static bool display_symbol_table = false;
|
|
static bool display_dynamic_symbol_table = false;
|
|
static bool display_core_notes = false;
|
|
static bool display_relocations = false;
|
|
static bool display_unwind_info = false;
|
|
static bool display_dynamic_section = false;
|
|
static bool display_hardening = false;
|
|
|
|
Core::ArgsParser args_parser;
|
|
args_parser.add_option(display_all, "Display all", "all", 'a');
|
|
args_parser.add_option(display_elf_header, "Display ELF header", "file-header", 'h');
|
|
args_parser.add_option(display_program_headers, "Display program headers", "program-headers", 'l');
|
|
args_parser.add_option(display_section_headers, "Display section headers", "section-headers", 'S');
|
|
args_parser.add_option(display_headers, "Equivalent to: -h -l -S -s -r -d -n -u -c", "headers", 'e');
|
|
args_parser.add_option(display_symbol_table, "Display the symbol table", "syms", 's');
|
|
args_parser.add_option(display_dynamic_symbol_table, "Display the dynamic symbol table", "dyn-syms", '\0');
|
|
args_parser.add_option(display_dynamic_section, "Display the dynamic section", "dynamic", 'd');
|
|
args_parser.add_option(display_core_notes, "Display core notes", "notes", 'n');
|
|
args_parser.add_option(display_relocations, "Display relocations", "relocs", 'r');
|
|
args_parser.add_option(display_unwind_info, "Display unwind info", "unwind", 'u');
|
|
args_parser.add_option(display_hardening, "Display security hardening info", "checksec", 'c');
|
|
args_parser.add_positional_argument(path, "ELF path", "path");
|
|
args_parser.parse(argc, argv);
|
|
|
|
if (argc < 3) {
|
|
args_parser.print_usage(stderr, argv[0]);
|
|
return -1;
|
|
}
|
|
|
|
if (display_headers) {
|
|
display_elf_header = true;
|
|
display_program_headers = true;
|
|
display_section_headers = true;
|
|
}
|
|
|
|
if (display_all) {
|
|
display_elf_header = true;
|
|
display_program_headers = true;
|
|
display_section_headers = true;
|
|
display_core_notes = true;
|
|
display_relocations = true;
|
|
display_unwind_info = true;
|
|
display_symbol_table = true;
|
|
display_hardening = true;
|
|
}
|
|
|
|
auto file_or_error = MappedFile::map(path);
|
|
|
|
if (file_or_error.is_error()) {
|
|
warnln("Unable to map file {}: {}", path, file_or_error.error());
|
|
return -1;
|
|
}
|
|
|
|
auto elf_image_data = file_or_error.value()->bytes();
|
|
ELF::Image elf_image(elf_image_data);
|
|
|
|
if (!elf_image.is_valid()) {
|
|
fprintf(stderr, "File is not a valid ELF object\n");
|
|
return -1;
|
|
}
|
|
|
|
String interpreter_path;
|
|
|
|
if (!ELF::validate_program_headers(*(const Elf32_Ehdr*)elf_image_data.data(), elf_image_data.size(), (const u8*)elf_image_data.data(), elf_image_data.size(), &interpreter_path)) {
|
|
fprintf(stderr, "Invalid ELF headers\n");
|
|
return -1;
|
|
}
|
|
|
|
auto& header = *reinterpret_cast<const Elf32_Ehdr*>(elf_image_data.data());
|
|
|
|
RefPtr<ELF::DynamicObject> object = nullptr;
|
|
|
|
if (elf_image.is_dynamic()) {
|
|
if (interpreter_path.is_null()) {
|
|
interpreter_path = "/usr/lib/Loader.so";
|
|
fprintf(stderr, "Warning: Dynamic ELF object has no interpreter path. Using: %s\n", interpreter_path.characters());
|
|
}
|
|
|
|
auto interpreter_file_or_error = MappedFile::map(interpreter_path);
|
|
|
|
if (interpreter_file_or_error.is_error()) {
|
|
warnln("Unable to map interpreter file {}: {}", interpreter_path, interpreter_file_or_error.error());
|
|
return -1;
|
|
}
|
|
|
|
auto interpreter_image_data = interpreter_file_or_error.value()->bytes();
|
|
|
|
ELF::Image interpreter_image(interpreter_image_data);
|
|
|
|
if (!interpreter_image.is_valid()) {
|
|
fprintf(stderr, "ELF interpreter image is invalid\n");
|
|
return -1;
|
|
}
|
|
|
|
int fd = open(path, O_RDONLY);
|
|
if (fd < 0) {
|
|
outln("Unable to open file {}", path);
|
|
return 1;
|
|
}
|
|
|
|
auto result = ELF::DynamicLoader::try_create(fd, path);
|
|
if (result.is_error()) {
|
|
outln("{}", result.error().text);
|
|
return 1;
|
|
}
|
|
auto& loader = result.value();
|
|
if (!loader->is_valid()) {
|
|
outln("{} is not a valid ELF dynamic shared object!", path);
|
|
return 1;
|
|
}
|
|
|
|
object = loader->map();
|
|
if (!object) {
|
|
outln("Failed to map dynamic ELF object {}", path);
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
if (display_elf_header) {
|
|
printf("ELF header:\n");
|
|
|
|
printf(" Magic: ");
|
|
for (char i : StringView { header.e_ident, sizeof(header.e_ident) }) {
|
|
if (isprint(i)) {
|
|
printf("%c ", i);
|
|
} else {
|
|
printf("%02x ", i);
|
|
}
|
|
}
|
|
printf("\n");
|
|
|
|
printf(" Type: %d (%s)\n", header.e_type, object_file_type_to_string(header.e_type));
|
|
printf(" Machine: %u (%s)\n", header.e_machine, object_machine_type_to_string(header.e_machine));
|
|
printf(" Version: 0x%x\n", header.e_version);
|
|
printf(" Entry point address: 0x%x\n", header.e_entry);
|
|
printf(" Start of program headers: %u (bytes into file)\n", header.e_phoff);
|
|
printf(" Start of section headers: %u (bytes into file)\n", header.e_shoff);
|
|
printf(" Flags: 0x%x\n", header.e_flags);
|
|
printf(" Size of this header: %u (bytes)\n", header.e_ehsize);
|
|
printf(" Size of program headers: %u (bytes)\n", header.e_phentsize);
|
|
printf(" Number of program headers: %u\n", header.e_phnum);
|
|
printf(" Size of section headers: %u (bytes)\n", header.e_shentsize);
|
|
printf(" Number of section headers: %u\n", header.e_shnum);
|
|
printf(" Section header string table index: %u\n", header.e_shstrndx);
|
|
printf("\n");
|
|
}
|
|
|
|
if (display_section_headers) {
|
|
if (!display_all) {
|
|
printf("There are %u section headers, starting at offset 0x%x:\n", header.e_shnum, header.e_shoff);
|
|
printf("\n");
|
|
}
|
|
|
|
if (!elf_image.section_count()) {
|
|
printf("There are no sections in this file.\n");
|
|
} else {
|
|
printf("Section Headers:\n");
|
|
printf(" Name Type Address Offset Size Flags\n");
|
|
|
|
elf_image.for_each_section([](const ELF::Image::Section& section) {
|
|
printf(" %-19s ", StringView(section.name()).to_string().characters());
|
|
printf("%-15s ", object_section_header_type_to_string(section.type()));
|
|
printf("%08x ", section.address());
|
|
printf("%08x ", section.offset());
|
|
printf("%08x ", section.size());
|
|
printf("%u", section.flags());
|
|
printf("\n");
|
|
return IterationDecision::Continue;
|
|
});
|
|
}
|
|
printf("\n");
|
|
}
|
|
|
|
if (display_program_headers) {
|
|
if (!display_all) {
|
|
printf("Elf file type is %d (%s)\n", header.e_type, object_file_type_to_string(header.e_type));
|
|
printf("Entry point 0x%x\n", header.e_entry);
|
|
printf("There are %u program headers, starting at offset %u\n", header.e_phnum, header.e_phoff);
|
|
printf("\n");
|
|
}
|
|
|
|
if (!elf_image.program_header_count()) {
|
|
printf("There are no program headers in this file.\n");
|
|
} else {
|
|
printf("Program Headers:\n");
|
|
printf(" Type Offset VirtAddr PhysAddr FileSiz MemSiz Flg Align\n");
|
|
|
|
elf_image.for_each_program_header([](const ELF::Image::ProgramHeader& program_header) {
|
|
printf(" %-14s ", object_program_header_type_to_string(program_header.type()));
|
|
printf("0x%08x ", program_header.offset());
|
|
printf("%p ", program_header.vaddr().as_ptr());
|
|
printf("%p ", program_header.vaddr().as_ptr()); // FIXME: assumes PhysAddr = VirtAddr
|
|
printf("0x%08x ", program_header.size_in_image());
|
|
printf("0x%08x ", program_header.size_in_memory());
|
|
printf("%04x ", program_header.flags());
|
|
printf("0x%08x", program_header.alignment());
|
|
printf("\n");
|
|
|
|
if (program_header.type() == PT_INTERP)
|
|
printf(" [Interpreter: %s]\n", program_header.raw_data());
|
|
|
|
return IterationDecision::Continue;
|
|
});
|
|
}
|
|
|
|
// TODO: Display section to segment mapping
|
|
printf("\n");
|
|
}
|
|
|
|
if (display_dynamic_section) {
|
|
auto found_dynamic_section = false;
|
|
if (elf_image.is_dynamic()) {
|
|
elf_image.for_each_section([&found_dynamic_section](const ELF::Image::Section& section) {
|
|
if (section.name() != ELF_DYNAMIC)
|
|
return IterationDecision::Continue;
|
|
|
|
found_dynamic_section = true;
|
|
|
|
if (section.entry_count()) {
|
|
printf("Dynamic section '%s' at offset 0x%08x contains %u entries.\n", section.name().to_string().characters(), section.offset(), section.entry_count());
|
|
} else {
|
|
printf("Dynamic section '%s' at offset 0x%08x contains zero entries.\n", section.name().to_string().characters(), section.offset());
|
|
}
|
|
|
|
return IterationDecision::Break;
|
|
});
|
|
|
|
Vector<String> libraries;
|
|
object->for_each_needed_library([&libraries](StringView entry) {
|
|
libraries.append(String::formatted("{}", entry).characters());
|
|
return IterationDecision::Continue;
|
|
});
|
|
|
|
auto library_index = 0;
|
|
printf(" Tag Type Name / Value\n");
|
|
object->for_each_dynamic_entry([&library_index, &libraries, &object](const ELF::DynamicObject::DynamicEntry& entry) {
|
|
printf(" 0x%08x ", entry.tag());
|
|
printf("%-17s ", object_tag_to_string(entry.tag()));
|
|
|
|
if (entry.tag() == DT_NEEDED) {
|
|
printf("Shared library: %s\n", String(libraries[library_index]).characters());
|
|
library_index++;
|
|
} else if (entry.tag() == DT_RPATH) {
|
|
printf("Library rpath: %s\n", String(object->rpath()).characters());
|
|
} else if (entry.tag() == DT_RUNPATH) {
|
|
printf("Library runpath: %s\n", String(object->runpath()).characters());
|
|
} else if (entry.tag() == DT_SONAME) {
|
|
printf("Library soname: %s\n", String(object->soname()).characters());
|
|
} else {
|
|
printf("0x%08x\n", entry.val());
|
|
}
|
|
return IterationDecision::Continue;
|
|
});
|
|
}
|
|
|
|
if (!found_dynamic_section)
|
|
printf("No dynamic section in this file.\n");
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
if (display_relocations) {
|
|
if (elf_image.is_dynamic()) {
|
|
if (!object->relocation_section().entry_count()) {
|
|
printf("Relocation section '%s' at offset 0x%08x contains zero entries:\n", object->relocation_section().name().to_string().characters(), object->relocation_section().offset());
|
|
} else {
|
|
printf("Relocation section '%s' at offset 0x%08x contains %u entries:\n", object->relocation_section().name().to_string().characters(), object->relocation_section().offset(), object->relocation_section().entry_count());
|
|
printf(" Offset Type Sym Value Sym Name\n");
|
|
object->relocation_section().for_each_relocation([](const ELF::DynamicObject::Relocation& reloc) {
|
|
printf(" 0x%08x ", reloc.offset());
|
|
printf(" %-17s ", object_relocation_type_to_string(reloc.type()));
|
|
printf(" 0x%08x ", reloc.symbol().value());
|
|
printf(" %s", reloc.symbol().name().to_string().characters());
|
|
printf("\n");
|
|
return IterationDecision::Continue;
|
|
});
|
|
}
|
|
printf("\n");
|
|
|
|
if (!object->plt_relocation_section().entry_count()) {
|
|
printf("Relocation section '%s' at offset 0x%08x contains zero entries:\n", object->plt_relocation_section().name().to_string().characters(), object->plt_relocation_section().offset());
|
|
} else {
|
|
printf("Relocation section '%s' at offset 0x%08x contains %u entries:\n", object->plt_relocation_section().name().to_string().characters(), object->plt_relocation_section().offset(), object->plt_relocation_section().entry_count());
|
|
printf(" Offset Type Sym Value Sym Name\n");
|
|
object->plt_relocation_section().for_each_relocation([](const ELF::DynamicObject::Relocation& reloc) {
|
|
printf(" 0x%08x ", reloc.offset());
|
|
printf(" %-17s ", object_relocation_type_to_string(reloc.type()));
|
|
printf(" 0x%08x ", reloc.symbol().value());
|
|
printf(" %s", reloc.symbol().name().to_string().characters());
|
|
printf("\n");
|
|
return IterationDecision::Continue;
|
|
});
|
|
}
|
|
} else {
|
|
printf("No relocations in this file.\n");
|
|
}
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
if (display_unwind_info) {
|
|
// TODO: Unwind info
|
|
printf("Decoding of unwind sections for machine type %s is not supported.\n", object_machine_type_to_string(header.e_machine));
|
|
printf("\n");
|
|
}
|
|
|
|
if (display_core_notes) {
|
|
auto found_notes = false;
|
|
elf_image.for_each_program_header([&found_notes](const ELF::Image::ProgramHeader& program_header) {
|
|
if (program_header.type() != PT_NOTE)
|
|
return IterationDecision::Continue;
|
|
|
|
found_notes = true;
|
|
|
|
printf("Displaying notes section '%s' at offset 0x%08x of length 0x%08x:\n", object_program_header_type_to_string(program_header.type()), program_header.offset(), program_header.size_in_image());
|
|
|
|
// FIXME: Parse CORE notes. Notes are in JSON format on SerenityOS, but vary between systems.
|
|
printf("%s\n", program_header.raw_data());
|
|
|
|
return IterationDecision::Continue;
|
|
});
|
|
|
|
if (!found_notes)
|
|
printf("No core notes in this file.\n");
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
if (display_dynamic_symbol_table || display_symbol_table) {
|
|
auto found_dynamic_symbol_table = false;
|
|
|
|
if (elf_image.is_dynamic()) {
|
|
elf_image.for_each_section([&found_dynamic_symbol_table](const ELF::Image::Section& section) {
|
|
if (section.name() != ELF_DYNSYM)
|
|
return IterationDecision::Continue;
|
|
|
|
found_dynamic_symbol_table = true;
|
|
|
|
if (!section.entry_count()) {
|
|
printf("Symbol table '%s' contains zero entries.\n", ELF_DYNSYM);
|
|
} else {
|
|
printf("Symbol table '%s' contains %u entries.\n", ELF_DYNSYM, section.entry_count());
|
|
}
|
|
|
|
return IterationDecision::Break;
|
|
});
|
|
|
|
if (object->symbol_count()) {
|
|
// FIXME: Add support for init/fini/start/main sections
|
|
printf(" Num: Value Size Type Bind Name\n");
|
|
object->for_each_symbol([](const ELF::DynamicObject::Symbol& sym) {
|
|
printf(" %4u: ", sym.index());
|
|
printf("%08x ", sym.value());
|
|
printf("%08x ", sym.size());
|
|
printf("%-8s ", object_symbol_type_to_string(sym.type()));
|
|
printf("%-8s ", object_symbol_binding_to_string(sym.bind()));
|
|
printf("%s", StringView(sym.name()).to_string().characters());
|
|
printf("\n");
|
|
return IterationDecision::Continue;
|
|
});
|
|
}
|
|
}
|
|
|
|
if (!found_dynamic_symbol_table)
|
|
printf("No dynamic symbol information for this file.\n");
|
|
|
|
printf("\n");
|
|
}
|
|
|
|
if (display_symbol_table) {
|
|
if (elf_image.symbol_count()) {
|
|
printf("Symbol table '%s' contains %u entries:\n", ELF_SYMTAB, elf_image.symbol_count());
|
|
printf(" Num: Value Size Type Bind Name\n");
|
|
|
|
elf_image.for_each_symbol([](const ELF::Image::Symbol& sym) {
|
|
printf(" %4u: ", sym.index());
|
|
printf("%08x ", sym.value());
|
|
printf("%08x ", sym.size());
|
|
printf("%-8s ", object_symbol_type_to_string(sym.type()));
|
|
printf("%-8s ", object_symbol_binding_to_string(sym.bind()));
|
|
printf("%s", StringView(sym.name()).to_string().characters());
|
|
printf("\n");
|
|
return IterationDecision::Continue;
|
|
});
|
|
} else {
|
|
printf("Symbol table '%s' contains zero entries.\n", ELF_SYMTAB);
|
|
}
|
|
printf("\n");
|
|
}
|
|
|
|
if (display_hardening) {
|
|
printf("Security Hardening:\n");
|
|
printf("RELRO Stack Canary NX PIE RPATH RUNPATH Symbols \n");
|
|
|
|
bool relro = false;
|
|
elf_image.for_each_program_header([&relro](const ELF::Image::ProgramHeader& program_header) {
|
|
if (program_header.type() == PT_GNU_RELRO) {
|
|
relro = true;
|
|
return IterationDecision::Break;
|
|
}
|
|
return IterationDecision::Continue;
|
|
});
|
|
|
|
bool full_relro = false;
|
|
if (relro) {
|
|
object->for_each_dynamic_entry([&full_relro](const ELF::DynamicObject::DynamicEntry& entry) {
|
|
if (entry.tag() == DT_BIND_NOW) {
|
|
full_relro = true;
|
|
return IterationDecision::Break;
|
|
}
|
|
return IterationDecision::Continue;
|
|
});
|
|
if (full_relro)
|
|
printf("\033[0;32m%-13s\033[0m ", "Full RELRO");
|
|
else
|
|
printf("\033[0;33m%-13s\033[0m ", "Partial RELRO");
|
|
} else {
|
|
printf("\033[0;31m%-13s\033[0m ", "No RELRO");
|
|
}
|
|
|
|
bool canary = false;
|
|
elf_image.for_each_symbol([&canary](const ELF::Image::Symbol& sym) {
|
|
if (sym.name() == "__stack_chk_fail" || sym.name() == "__intel_security_cookie") {
|
|
canary = true;
|
|
return IterationDecision::Break;
|
|
}
|
|
return IterationDecision::Continue;
|
|
});
|
|
|
|
if (canary)
|
|
printf("\033[0;32m%-12s\033[0m ", "Canary found");
|
|
else
|
|
printf("\033[0;31m%-12s\033[0m ", "No canary");
|
|
|
|
bool nx = false;
|
|
elf_image.for_each_program_header([&nx](const ELF::Image::ProgramHeader& program_header) {
|
|
if (program_header.type() == PT_GNU_STACK) {
|
|
if (program_header.flags() & PF_X)
|
|
nx = false;
|
|
else
|
|
nx = true;
|
|
return IterationDecision::Break;
|
|
}
|
|
return IterationDecision::Continue;
|
|
});
|
|
|
|
if (nx)
|
|
printf("\033[0;32m%-12s\033[0m ", "NX enabled");
|
|
else
|
|
printf("\033[0;31m%-12s\033[0m ", "NX disabled");
|
|
|
|
bool pie = false;
|
|
if (header.e_type == ET_REL || header.e_type == ET_DYN)
|
|
pie = true;
|
|
|
|
if (pie)
|
|
printf("\033[0;32m%-12s\033[0m ", "PIE enabled");
|
|
else
|
|
printf("\033[0;31m%-12s\033[0m ", "No PIE");
|
|
|
|
StringView rpath;
|
|
if (elf_image.is_dynamic())
|
|
rpath = object->rpath();
|
|
|
|
if (rpath.is_empty())
|
|
printf("\033[0;32m%-12s\033[0m ", "No RPATH");
|
|
else
|
|
printf("\033[0;31m%-12s\033[0m ", rpath.to_string().characters());
|
|
|
|
StringView runpath;
|
|
if (elf_image.is_dynamic())
|
|
runpath = object->runpath();
|
|
|
|
if (runpath.is_empty())
|
|
printf("\033[0;32m%-12s\033[0m ", "No RUNPATH");
|
|
else
|
|
printf("\033[0;31m%-12s\033[0m ", runpath.to_string().characters());
|
|
|
|
printf("%u symbols ", elf_image.symbol_count());
|
|
printf("\n");
|
|
}
|
|
|
|
return 0;
|
|
}
|