LibELF: Use StringBuilders instead of Strings for the interpreter path

This is required for the Kernel's usage of LibELF, since Strings do not
expose allocation failure.
This commit is contained in:
Idan Horowitz 2022-01-13 18:50:17 +02:00 committed by Andreas Kling
parent fb3e46e930
commit 3e959618c3
Notes: sideshowbarker 2024-07-17 20:56:12 +09:00
8 changed files with 36 additions and 18 deletions

View File

@ -713,11 +713,12 @@ static ErrorOr<NonnullOwnPtrVector<KString>> find_shebang_interpreter_for_execut
ErrorOr<RefPtr<OpenFileDescription>> Process::find_elf_interpreter_for_executable(StringView path, ElfW(Ehdr) const& main_executable_header, size_t main_executable_header_size, size_t file_size) ErrorOr<RefPtr<OpenFileDescription>> Process::find_elf_interpreter_for_executable(StringView path, ElfW(Ehdr) const& main_executable_header, size_t main_executable_header_size, size_t file_size)
{ {
// Not using ErrorOr here because we'll want to do the same thing in userspace in the RTLD // Not using ErrorOr here because we'll want to do the same thing in userspace in the RTLD
String interpreter_path; StringBuilder interpreter_path_builder;
if (!ELF::validate_program_headers(main_executable_header, file_size, (u8 const*)&main_executable_header, main_executable_header_size, &interpreter_path)) { if (!TRY(ELF::validate_program_headers(main_executable_header, file_size, (u8 const*)&main_executable_header, main_executable_header_size, &interpreter_path_builder))) {
dbgln("exec({}): File has invalid ELF Program headers", path); dbgln("exec({}): File has invalid ELF Program headers", path);
return ENOEXEC; return ENOEXEC;
} }
auto interpreter_path = interpreter_path_builder.string_view();
if (!interpreter_path.is_empty()) { if (!interpreter_path.is_empty()) {
dbgln_if(EXEC_DEBUG, "exec({}): Using program interpreter {}", path, interpreter_path); dbgln_if(EXEC_DEBUG, "exec({}): Using program interpreter {}", path, interpreter_path);
@ -745,11 +746,12 @@ ErrorOr<RefPtr<OpenFileDescription>> Process::find_elf_interpreter_for_executabl
} }
// Not using ErrorOr here because we'll want to do the same thing in userspace in the RTLD // Not using ErrorOr here because we'll want to do the same thing in userspace in the RTLD
String interpreter_interpreter_path; StringBuilder interpreter_interpreter_path_builder;
if (!ELF::validate_program_headers(*elf_header, interp_metadata.size, (u8*)first_page, nread, &interpreter_interpreter_path)) { if (!TRY(ELF::validate_program_headers(*elf_header, interp_metadata.size, (u8*)first_page, nread, &interpreter_interpreter_path_builder))) {
dbgln("exec({}): Interpreter ({}) has invalid ELF Program headers", path, interpreter_path); dbgln("exec({}): Interpreter ({}) has invalid ELF Program headers", path, interpreter_path);
return ENOEXEC; return ENOEXEC;
} }
auto interpreter_interpreter_path = interpreter_interpreter_path_builder.string_view();
if (!interpreter_interpreter_path.is_empty()) { if (!interpreter_interpreter_path.is_empty()) {
dbgln("exec({}): Interpreter ({}) has its own interpreter ({})! No thank you!", path, interpreter_path, interpreter_interpreter_path); dbgln("exec({}): Interpreter ({}) has its own interpreter ({})! No thank you!", path, interpreter_path, interpreter_interpreter_path);

View File

@ -166,11 +166,13 @@ bool Emulator::load_elf()
VERIFY_NOT_REACHED(); VERIFY_NOT_REACHED();
} }
String interpreter_path; StringBuilder interpreter_path_builder;
if (!ELF::validate_program_headers(*(Elf32_Ehdr const*)elf_image_data.data(), elf_image_data.size(), (u8 const*)elf_image_data.data(), elf_image_data.size(), &interpreter_path)) { auto result_or_error = ELF::validate_program_headers(*(Elf32_Ehdr const*)elf_image_data.data(), elf_image_data.size(), (u8 const*)elf_image_data.data(), elf_image_data.size(), &interpreter_path_builder);
if (result_or_error.is_error() || !result_or_error.value()) {
reportln("failed to validate ELF file"); reportln("failed to validate ELF file");
return false; return false;
} }
auto interpreter_path = interpreter_path_builder.string_view();
VERIFY(!interpreter_path.is_null()); VERIFY(!interpreter_path.is_null());
dbgln("interpreter: {}", interpreter_path); dbgln("interpreter: {}", interpreter_path);

View File

@ -122,8 +122,11 @@ bool DynamicLoader::validate()
auto* elf_header = (ElfW(Ehdr)*)m_file_data; auto* elf_header = (ElfW(Ehdr)*)m_file_data;
if (!validate_elf_header(*elf_header, m_file_size)) if (!validate_elf_header(*elf_header, m_file_size))
return false; return false;
if (!validate_program_headers(*elf_header, m_file_size, (u8*)m_file_data, m_file_size, &m_program_interpreter)) StringBuilder interpreter_path_builder;
auto result_or_error = validate_program_headers(*elf_header, m_file_size, (u8*)m_file_data, m_file_size, &interpreter_path_builder);
if (result_or_error.is_error() || !result_or_error.value())
return false; return false;
m_program_interpreter = interpreter_path_builder.string_view();
return true; return true;
} }

View File

@ -126,7 +126,14 @@ bool Image::parse()
return false; return false;
} }
if (!validate_program_headers(header(), m_size, m_buffer, m_size, nullptr, m_verbose_logging)) { auto result_or_error = validate_program_headers(header(), m_size, m_buffer, m_size, nullptr, m_verbose_logging);
if (result_or_error.is_error()) {
if (m_verbose_logging)
dbgln("ELF::Image::parse(): Failed validating ELF Program Headers");
m_valid = false;
return false;
}
if (!result_or_error.value()) {
if (m_verbose_logging) if (m_verbose_logging)
dbgln("ELF::Image::parse(): ELF Program Headers not valid"); dbgln("ELF::Image::parse(): ELF Program Headers not valid");
m_valid = false; m_valid = false;

View File

@ -192,7 +192,7 @@ bool validate_elf_header(const ElfW(Ehdr) & elf_header, size_t file_size, bool v
return true; return true;
} }
bool validate_program_headers(const ElfW(Ehdr) & elf_header, size_t file_size, const u8* buffer, size_t buffer_size, String* interpreter_path, bool verbose) ErrorOr<bool> validate_program_headers(const ElfW(Ehdr) & elf_header, size_t file_size, const u8* buffer, size_t buffer_size, StringBuilder* interpreter_path_builder, bool verbose)
{ {
Checked<size_t> total_size_of_program_headers = elf_header.e_phnum; Checked<size_t> total_size_of_program_headers = elf_header.e_phnum;
total_size_of_program_headers *= elf_header.e_phentsize; total_size_of_program_headers *= elf_header.e_phentsize;
@ -268,8 +268,8 @@ bool validate_program_headers(const ElfW(Ehdr) & elf_header, size_t file_size, c
dbgln("Found PT_INTERP header ({}), but p_filesz is invalid ({})", header_index, program_header.p_filesz); dbgln("Found PT_INTERP header ({}), but p_filesz is invalid ({})", header_index, program_header.p_filesz);
return false; return false;
} }
if (interpreter_path) if (interpreter_path_builder)
*interpreter_path = String((const char*)&buffer[program_header.p_offset], program_header.p_filesz - 1); TRY(interpreter_path_builder->try_append({ &buffer[program_header.p_offset], program_header.p_filesz - 1 }));
break; break;
case PT_LOAD: case PT_LOAD:
case PT_DYNAMIC: case PT_DYNAMIC:

View File

@ -8,10 +8,11 @@
#include <AK/String.h> #include <AK/String.h>
#include <LibC/elf.h> #include <LibC/elf.h>
#include <limits.h>
namespace ELF { namespace ELF {
bool validate_elf_header(const ElfW(Ehdr) & elf_header, size_t file_size, bool verbose = true); bool validate_elf_header(const ElfW(Ehdr) & elf_header, size_t file_size, bool verbose = true);
bool validate_program_headers(const ElfW(Ehdr) & elf_header, size_t file_size, const u8* buffer, size_t buffer_size, String* interpreter_path, bool verbose = true); ErrorOr<bool> validate_program_headers(const ElfW(Ehdr) & elf_header, size_t file_size, const u8* buffer, size_t buffer_size, StringBuilder* interpreter_path_builder, bool verbose = true);
} // end namespace ELF } // end namespace ELF

View File

@ -67,9 +67,11 @@ static Optional<String> elf_details(String description, const String& path)
if (!elf_image.is_valid()) if (!elf_image.is_valid())
return {}; return {};
String interpreter_path; StringBuilder interpreter_path_builder;
if (!ELF::validate_program_headers(*(const ElfW(Ehdr)*)elf_data.data(), elf_data.size(), (const u8*)elf_data.data(), elf_data.size(), &interpreter_path)) auto result_or_error = ELF::validate_program_headers(*(const ElfW(Ehdr)*)elf_data.data(), elf_data.size(), (const u8*)elf_data.data(), elf_data.size(), &interpreter_path_builder);
if (result_or_error.is_error() || !result_or_error.value())
return {}; return {};
auto interpreter_path = interpreter_path_builder.string_view();
auto& header = *reinterpret_cast<const ElfW(Ehdr)*>(elf_data.data()); auto& header = *reinterpret_cast<const ElfW(Ehdr)*>(elf_data.data());

View File

@ -298,12 +298,13 @@ int main(int argc, char** argv)
return -1; return -1;
} }
String interpreter_path; StringBuilder interpreter_path_builder;
auto result_or_error = ELF::validate_program_headers(*(const ElfW(Ehdr)*)elf_image_data.data(), elf_image_data.size(), (const u8*)elf_image_data.data(), elf_image_data.size(), &interpreter_path_builder);
if (!ELF::validate_program_headers(*(const ElfW(Ehdr)*)elf_image_data.data(), elf_image_data.size(), (const u8*)elf_image_data.data(), elf_image_data.size(), &interpreter_path)) { if (result_or_error.is_error() || !result_or_error.value()) {
warnln("Invalid ELF headers"); warnln("Invalid ELF headers");
return -1; return -1;
} }
auto interpreter_path = interpreter_path_builder.string_view();
auto& header = *reinterpret_cast<const ElfW(Ehdr)*>(elf_image_data.data()); auto& header = *reinterpret_cast<const ElfW(Ehdr)*>(elf_image_data.data());
@ -311,7 +312,7 @@ int main(int argc, char** argv)
if (elf_image.is_dynamic()) { if (elf_image.is_dynamic()) {
if (interpreter_path.is_null()) { if (interpreter_path.is_null()) {
interpreter_path = "/usr/lib/Loader.so"; interpreter_path = "/usr/lib/Loader.so"sv;
warnln("Warning: Dynamic ELF object has no interpreter path. Using: {}", interpreter_path); warnln("Warning: Dynamic ELF object has no interpreter path. Using: {}", interpreter_path);
} }