Kernel: Only load program headers of interpreter if its present
I was loading program headers of both executable and interpreter but that is incorrect. The interpreter will itself load the program headers of the executable.
This commit is contained in:
parent
50ab391133
commit
8da2f12ba6
|
@ -519,8 +519,8 @@ namespace Kernel
|
||||||
{
|
{
|
||||||
VirtualFileSystem::File file;
|
VirtualFileSystem::File file;
|
||||||
TRY(file.canonical_path.append("<self>"));
|
TRY(file.canonical_path.append("<self>"));
|
||||||
file.inode = m_loadable_elf->inode();
|
file.inode = m_loadable_elf->executable();
|
||||||
m_userspace_info.file_fd = TRY(m_open_file_descriptors.open(BAN::move(file), O_EXEC));
|
m_userspace_info.file_fd = TRY(m_open_file_descriptors.open(BAN::move(file), O_RDONLY));
|
||||||
}
|
}
|
||||||
|
|
||||||
for (size_t i = 0; i < sizeof(m_signal_handlers) / sizeof(*m_signal_handlers); i++)
|
for (size_t i = 0; i < sizeof(m_signal_handlers) / sizeof(*m_signal_handlers); i++)
|
||||||
|
|
|
@ -26,26 +26,20 @@ namespace LibELF
|
||||||
|
|
||||||
LoadableELF::~LoadableELF()
|
LoadableELF::~LoadableELF()
|
||||||
{
|
{
|
||||||
const auto cleanup_program_headers =
|
|
||||||
[&](BAN::Span<const ElfNativeProgramHeader> headers)
|
|
||||||
{
|
|
||||||
for (const auto& header : headers)
|
|
||||||
{
|
|
||||||
ASSERT(header.p_type == PT_LOAD);
|
|
||||||
|
|
||||||
const vaddr_t vaddr = header.p_vaddr & PAGE_ADDR_MASK;
|
|
||||||
const size_t pages = range_page_count(header.p_vaddr, header.p_memsz);
|
|
||||||
for (size_t i = 0; i < pages; i++)
|
|
||||||
if (paddr_t paddr = m_page_table.physical_address_of(vaddr + i * PAGE_SIZE))
|
|
||||||
Heap::get().release_page(paddr);
|
|
||||||
m_page_table.unmap_range(vaddr, pages * PAGE_SIZE);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if (!m_is_loaded)
|
if (!m_is_loaded)
|
||||||
return;
|
return;
|
||||||
cleanup_program_headers(m_executable.program_headers.span());
|
|
||||||
cleanup_program_headers(m_interpreter.program_headers.span());
|
for (const auto& header : m_program_headers)
|
||||||
|
{
|
||||||
|
ASSERT(header.p_type == PT_LOAD);
|
||||||
|
|
||||||
|
const vaddr_t vaddr = header.p_vaddr & PAGE_ADDR_MASK;
|
||||||
|
const size_t pages = range_page_count(header.p_vaddr, header.p_memsz);
|
||||||
|
for (size_t i = 0; i < pages; i++)
|
||||||
|
if (paddr_t paddr = m_page_table.physical_address_of(vaddr + i * PAGE_SIZE))
|
||||||
|
Heap::get().release_page(paddr);
|
||||||
|
m_page_table.unmap_range(vaddr, pages * PAGE_SIZE);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static BAN::ErrorOr<ElfNativeFileHeader> read_and_validate_file_header(BAN::RefPtr<Inode> inode)
|
static BAN::ErrorOr<ElfNativeFileHeader> read_and_validate_file_header(BAN::RefPtr<Inode> inode)
|
||||||
|
@ -166,30 +160,26 @@ namespace LibELF
|
||||||
}
|
}
|
||||||
|
|
||||||
return LoadResult {
|
return LoadResult {
|
||||||
.elf_file = {
|
.inode = inode,
|
||||||
.inode = inode,
|
.interp = interp,
|
||||||
.file_header = file_header,
|
.file_header = file_header,
|
||||||
.program_headers = BAN::move(program_headers),
|
.program_headers = BAN::move(program_headers)
|
||||||
.dynamic_base = 0
|
|
||||||
},
|
|
||||||
.interp = interp
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
bool LoadableELF::does_executable_and_interpreter_overlap() const
|
static bool do_program_headers_overlap(BAN::Span<const ElfNativeProgramHeader> pheaders1, BAN::Span<const ElfNativeProgramHeader> pheaders2, vaddr_t base2)
|
||||||
{
|
{
|
||||||
ASSERT(m_executable.inode);
|
for (const auto& pheader1 : pheaders1)
|
||||||
ASSERT(m_interpreter.inode);
|
|
||||||
|
|
||||||
for (const auto& epheader : m_executable.program_headers)
|
|
||||||
{
|
{
|
||||||
for (const auto& ipheader : m_interpreter.program_headers)
|
for (const auto& pheader2 : pheaders2)
|
||||||
{
|
{
|
||||||
const vaddr_t e1 = epheader.p_vaddr & PAGE_ADDR_MASK;
|
const vaddr_t s1 = pheader1.p_vaddr & PAGE_ADDR_MASK;
|
||||||
const vaddr_t i1 = ipheader.p_vaddr & PAGE_ADDR_MASK;
|
const vaddr_t e1 = (pheader1.p_vaddr + pheader1.p_memsz + PAGE_SIZE - 1) & PAGE_ADDR_MASK;
|
||||||
const vaddr_t e2 = (epheader.p_vaddr + epheader.p_memsz + PAGE_SIZE - 1) & PAGE_ADDR_MASK;
|
|
||||||
const vaddr_t i2 = (ipheader.p_vaddr + ipheader.p_memsz + PAGE_SIZE - 1) & PAGE_ADDR_MASK;
|
const vaddr_t s2 = pheader2.p_vaddr & PAGE_ADDR_MASK;
|
||||||
if (e1 < i2 && i1 < e2)
|
const vaddr_t e2 = (pheader2.p_vaddr + pheader2.p_memsz + PAGE_SIZE - 1) & PAGE_ADDR_MASK;
|
||||||
|
|
||||||
|
if (s1 < e2 + base2 && s2 + base2 < e1)
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -199,74 +189,95 @@ namespace LibELF
|
||||||
|
|
||||||
BAN::ErrorOr<void> LoadableELF::initialize(const Credentials& credentials, BAN::RefPtr<Inode> inode)
|
BAN::ErrorOr<void> LoadableELF::initialize(const Credentials& credentials, BAN::RefPtr<Inode> inode)
|
||||||
{
|
{
|
||||||
|
const auto generate_random_dynamic_base =
|
||||||
|
[]() -> vaddr_t
|
||||||
|
{
|
||||||
|
// 1 MiB -> 2 GiB + 1 MiB
|
||||||
|
return (Random::get_u32() & 0x7FFFF000) + 0x100000;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
auto executable_load_result = TRY(load_elf_file(credentials, inode));
|
auto executable_load_result = TRY(load_elf_file(credentials, inode));
|
||||||
m_executable = executable_load_result.elf_file;
|
|
||||||
|
|
||||||
if (m_executable.file_header.e_type == ET_DYN)
|
m_executable = executable_load_result.inode;
|
||||||
|
m_interpreter = executable_load_result.interp;
|
||||||
|
|
||||||
|
vaddr_t dynamic_base = 0;
|
||||||
|
|
||||||
|
if (m_interpreter)
|
||||||
{
|
{
|
||||||
m_executable.dynamic_base = (Random::get_u32() & 0x7FFFF000) + 0x100000;
|
auto interp_load_result = TRY(load_elf_file(credentials, m_interpreter));
|
||||||
m_executable.file_header.e_entry += m_executable.dynamic_base;
|
|
||||||
for (auto& program_header : m_executable.program_headers)
|
|
||||||
program_header.p_vaddr += m_executable.dynamic_base;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (executable_load_result.interp)
|
|
||||||
{
|
|
||||||
auto interp_load_result = TRY(load_elf_file(credentials, executable_load_result.interp));
|
|
||||||
m_interpreter = interp_load_result.elf_file;
|
|
||||||
|
|
||||||
if (interp_load_result.interp)
|
if (interp_load_result.interp)
|
||||||
{
|
{
|
||||||
dwarnln("Executable has specified interpreter for its interpreter");
|
dwarnln("ELF interpreter has an interpreter");
|
||||||
return BAN::Error::from_errno(EINVAL);
|
return BAN::Error::from_errno(EINVAL);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (m_interpreter.file_header.e_type == ET_DYN)
|
if (executable_load_result.file_header.e_type == ET_EXEC)
|
||||||
{
|
{
|
||||||
for (int attempt = 0; attempt < 100; attempt++)
|
if (interp_load_result.file_header.e_type == ET_EXEC)
|
||||||
{
|
{
|
||||||
const vaddr_t dynamic_base = (Random::get_u32() & 0x3FFFF000) + 0x40000000;
|
const bool has_overlap = do_program_headers_overlap(
|
||||||
for (auto& program_header : m_interpreter.program_headers)
|
executable_load_result.program_headers.span(),
|
||||||
program_header.p_vaddr += dynamic_base;
|
interp_load_result.program_headers.span(),
|
||||||
if (does_executable_and_interpreter_overlap())
|
0
|
||||||
|
);
|
||||||
|
|
||||||
|
if (has_overlap)
|
||||||
{
|
{
|
||||||
for (auto& program_header : m_interpreter.program_headers)
|
dwarnln("Executable and interpreter LOAD segments overlap");
|
||||||
program_header.p_vaddr -= dynamic_base;
|
return BAN::Error::from_errno(EINVAL);
|
||||||
continue;
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (int attempt = 0; attempt < 100; attempt++)
|
||||||
|
{
|
||||||
|
const vaddr_t test_dynamic_base = generate_random_dynamic_base();
|
||||||
|
const bool has_overlap = do_program_headers_overlap(
|
||||||
|
executable_load_result.program_headers.span(),
|
||||||
|
interp_load_result.program_headers.span(),
|
||||||
|
test_dynamic_base
|
||||||
|
);
|
||||||
|
if (has_overlap)
|
||||||
|
continue;
|
||||||
|
dynamic_base = test_dynamic_base;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dynamic_base == 0)
|
||||||
|
{
|
||||||
|
dwarnln("Could not find space to load interpreter");
|
||||||
|
return BAN::Error::from_errno(EINVAL);
|
||||||
}
|
}
|
||||||
m_interpreter.dynamic_base = dynamic_base;
|
|
||||||
m_interpreter.file_header.e_entry += dynamic_base;
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const bool can_load_interpreter = (m_interpreter.file_header.e_type == ET_DYN)
|
m_file_header = interp_load_result.file_header;
|
||||||
? (m_interpreter.dynamic_base != 0)
|
m_program_headers = BAN::move(interp_load_result.program_headers);
|
||||||
: !does_executable_and_interpreter_overlap();
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
m_file_header = executable_load_result.file_header;
|
||||||
|
m_program_headers = BAN::move(executable_load_result.program_headers);
|
||||||
|
}
|
||||||
|
|
||||||
if (!can_load_interpreter)
|
if (m_file_header.e_type == ET_DYN && dynamic_base == 0)
|
||||||
{
|
dynamic_base = generate_random_dynamic_base();
|
||||||
dwarnln("Could not find space to load interpreter");
|
|
||||||
return BAN::Error::from_errno(EINVAL);
|
if (dynamic_base)
|
||||||
}
|
{
|
||||||
|
m_file_header.e_entry += dynamic_base;
|
||||||
|
for (auto& program_header : m_program_headers)
|
||||||
|
program_header.p_vaddr += dynamic_base;
|
||||||
}
|
}
|
||||||
|
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
vaddr_t LoadableELF::entry_point() const
|
|
||||||
{
|
|
||||||
if (m_interpreter.inode)
|
|
||||||
return m_interpreter.file_header.e_entry;
|
|
||||||
return m_executable.file_header.e_entry;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool LoadableELF::contains(vaddr_t address) const
|
bool LoadableELF::contains(vaddr_t address) const
|
||||||
{
|
{
|
||||||
for (const auto& program_header : m_executable.program_headers)
|
for (const auto& program_header : m_program_headers)
|
||||||
if (program_header.p_vaddr <= address && address < program_header.p_vaddr + program_header.p_memsz)
|
|
||||||
return true;
|
|
||||||
for (const auto& program_header : m_interpreter.program_headers)
|
|
||||||
if (program_header.p_vaddr <= address && address < program_header.p_vaddr + program_header.p_memsz)
|
if (program_header.p_vaddr <= address && address < program_header.p_vaddr + program_header.p_memsz)
|
||||||
return true;
|
return true;
|
||||||
return false;
|
return false;
|
||||||
|
@ -274,112 +285,88 @@ namespace LibELF
|
||||||
|
|
||||||
bool LoadableELF::is_address_space_free() const
|
bool LoadableELF::is_address_space_free() const
|
||||||
{
|
{
|
||||||
const auto are_program_headers_free =
|
for (const auto& program_header : m_program_headers)
|
||||||
[&](BAN::Span<const ElfNativeProgramHeader> program_headers) -> bool
|
{
|
||||||
{
|
ASSERT(program_header.p_type == PT_LOAD);
|
||||||
for (const auto& program_header : program_headers)
|
const vaddr_t page_vaddr = program_header.p_vaddr & PAGE_ADDR_MASK;
|
||||||
{
|
const size_t pages = range_page_count(program_header.p_vaddr, program_header.p_memsz);
|
||||||
ASSERT(program_header.p_type == PT_LOAD);
|
if (!m_page_table.is_range_free(page_vaddr, pages * PAGE_SIZE))
|
||||||
const vaddr_t page_vaddr = program_header.p_vaddr & PAGE_ADDR_MASK;
|
return false;
|
||||||
const size_t pages = range_page_count(program_header.p_vaddr, program_header.p_memsz);
|
}
|
||||||
if (!m_page_table.is_range_free(page_vaddr, pages * PAGE_SIZE))
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
};
|
|
||||||
if (!are_program_headers_free(m_executable.program_headers.span()))
|
|
||||||
return false;
|
|
||||||
if (!are_program_headers_free(m_interpreter.program_headers.span()))
|
|
||||||
return false;
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void LoadableELF::reserve_address_space()
|
void LoadableELF::reserve_address_space()
|
||||||
{
|
{
|
||||||
const auto reserve_program_headers =
|
for (const auto& program_header : m_program_headers)
|
||||||
[&](BAN::Span<const ElfNativeProgramHeader> program_headers)
|
{
|
||||||
{
|
ASSERT(program_header.p_type == PT_LOAD);
|
||||||
for (const auto& program_header : program_headers)
|
const vaddr_t page_vaddr = program_header.p_vaddr & PAGE_ADDR_MASK;
|
||||||
{
|
const size_t pages = range_page_count(program_header.p_vaddr, program_header.p_memsz);
|
||||||
ASSERT(program_header.p_type == PT_LOAD);
|
if (!m_page_table.reserve_range(page_vaddr, pages * PAGE_SIZE))
|
||||||
const vaddr_t page_vaddr = program_header.p_vaddr & PAGE_ADDR_MASK;
|
ASSERT_NOT_REACHED();
|
||||||
const size_t pages = range_page_count(program_header.p_vaddr, program_header.p_memsz);
|
m_virtual_page_count += pages;
|
||||||
if (!m_page_table.reserve_range(page_vaddr, pages * PAGE_SIZE))
|
}
|
||||||
ASSERT_NOT_REACHED();
|
|
||||||
m_virtual_page_count += pages;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
reserve_program_headers(m_executable.program_headers.span());
|
|
||||||
reserve_program_headers(m_interpreter.program_headers.span());
|
|
||||||
m_is_loaded = true;
|
m_is_loaded = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void LoadableELF::update_suid_sgid(Kernel::Credentials& credentials)
|
void LoadableELF::update_suid_sgid(Kernel::Credentials& credentials)
|
||||||
{
|
{
|
||||||
auto inode = m_executable.inode;
|
if (m_executable->mode().mode & +Inode::Mode::ISUID)
|
||||||
ASSERT(inode);
|
credentials.set_euid(m_executable->uid());
|
||||||
|
if (m_executable->mode().mode & +Inode::Mode::ISGID)
|
||||||
if (inode->mode().mode & +Inode::Mode::ISUID)
|
credentials.set_egid(m_executable->gid());
|
||||||
credentials.set_euid(inode->uid());
|
|
||||||
if (inode->mode().mode & +Inode::Mode::ISGID)
|
|
||||||
credentials.set_egid(inode->gid());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
BAN::ErrorOr<void> LoadableELF::load_page_to_memory(vaddr_t address)
|
BAN::ErrorOr<void> LoadableELF::load_page_to_memory(vaddr_t address)
|
||||||
{
|
{
|
||||||
const auto load_page_from_program_header =
|
auto inode = has_interpreter() ? m_interpreter : m_executable;
|
||||||
[&](BAN::RefPtr<Inode> inode, BAN::Span<const ElfNativeProgramHeader> program_headers) -> BAN::ErrorOr<bool>
|
|
||||||
|
// FIXME: use MemoryBackedRegion/FileBackedRegion instead of manually mapping and allocating pages
|
||||||
|
|
||||||
|
for (const auto& program_header : m_program_headers)
|
||||||
|
{
|
||||||
|
ASSERT(program_header.p_type == PT_LOAD);
|
||||||
|
if (!(program_header.p_vaddr <= address && address < program_header.p_vaddr + program_header.p_memsz))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
PageTable::flags_t flags = PageTable::Flags::UserSupervisor | PageTable::Flags::Present;
|
||||||
|
if (program_header.p_flags & LibELF::PF_W)
|
||||||
|
flags |= PageTable::Flags::ReadWrite;
|
||||||
|
if (program_header.p_flags & LibELF::PF_X)
|
||||||
|
flags |= PageTable::Flags::Execute;
|
||||||
|
|
||||||
|
const vaddr_t vaddr = address & PAGE_ADDR_MASK;
|
||||||
|
const paddr_t paddr = Heap::get().take_free_page();
|
||||||
|
if (paddr == 0)
|
||||||
|
return BAN::Error::from_errno(ENOMEM);
|
||||||
|
|
||||||
|
// Temporarily map page as RW so kernel can write to it
|
||||||
|
m_page_table.map_page_at(paddr, vaddr, PageTable::Flags::ReadWrite | PageTable::Flags::Present);
|
||||||
|
m_physical_page_count++;
|
||||||
|
|
||||||
|
memset((void*)vaddr, 0x00, PAGE_SIZE);
|
||||||
|
|
||||||
|
if (vaddr / PAGE_SIZE < BAN::Math::div_round_up<size_t>(program_header.p_vaddr + program_header.p_filesz, PAGE_SIZE))
|
||||||
{
|
{
|
||||||
for (const auto& program_header : program_headers)
|
size_t vaddr_offset = 0;
|
||||||
{
|
if (vaddr < program_header.p_vaddr)
|
||||||
ASSERT(program_header.p_type == PT_LOAD);
|
vaddr_offset = program_header.p_vaddr - vaddr;
|
||||||
if (!(program_header.p_vaddr <= address && address < program_header.p_vaddr + program_header.p_memsz))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
PageTable::flags_t flags = PageTable::Flags::UserSupervisor | PageTable::Flags::Present;
|
size_t file_offset = 0;
|
||||||
if (program_header.p_flags & LibELF::PF_W)
|
if (vaddr > program_header.p_vaddr)
|
||||||
flags |= PageTable::Flags::ReadWrite;
|
file_offset = vaddr - program_header.p_vaddr;
|
||||||
if (program_header.p_flags & LibELF::PF_X)
|
|
||||||
flags |= PageTable::Flags::Execute;
|
|
||||||
|
|
||||||
const vaddr_t vaddr = address & PAGE_ADDR_MASK;
|
size_t bytes = BAN::Math::min<size_t>(PAGE_SIZE - vaddr_offset, program_header.p_filesz - file_offset);
|
||||||
const paddr_t paddr = Heap::get().take_free_page();
|
TRY(inode->read(program_header.p_offset + file_offset, { (uint8_t*)vaddr + vaddr_offset, bytes }));
|
||||||
if (paddr == 0)
|
}
|
||||||
return BAN::Error::from_errno(ENOMEM);
|
|
||||||
|
|
||||||
// Temporarily map page as RW so kernel can write to it
|
// Map page with the correct flags
|
||||||
m_page_table.map_page_at(paddr, vaddr, PageTable::Flags::ReadWrite | PageTable::Flags::Present);
|
m_page_table.map_page_at(paddr, vaddr, flags);
|
||||||
m_physical_page_count++;
|
|
||||||
|
|
||||||
memset((void*)vaddr, 0x00, PAGE_SIZE);
|
|
||||||
|
|
||||||
if (vaddr / PAGE_SIZE < BAN::Math::div_round_up<size_t>(program_header.p_vaddr + program_header.p_filesz, PAGE_SIZE))
|
|
||||||
{
|
|
||||||
size_t vaddr_offset = 0;
|
|
||||||
if (vaddr < program_header.p_vaddr)
|
|
||||||
vaddr_offset = program_header.p_vaddr - vaddr;
|
|
||||||
|
|
||||||
size_t file_offset = 0;
|
|
||||||
if (vaddr > program_header.p_vaddr)
|
|
||||||
file_offset = vaddr - program_header.p_vaddr;
|
|
||||||
|
|
||||||
size_t bytes = BAN::Math::min<size_t>(PAGE_SIZE - vaddr_offset, program_header.p_filesz - file_offset);
|
|
||||||
TRY(inode->read(program_header.p_offset + file_offset, { (uint8_t*)vaddr + vaddr_offset, bytes }));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Map page with the correct flags
|
|
||||||
m_page_table.map_page_at(paddr, vaddr, flags);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
};
|
|
||||||
|
|
||||||
if (TRY(load_page_from_program_header(m_executable.inode, m_executable.program_headers.span())))
|
|
||||||
return {};
|
|
||||||
if (TRY(load_page_from_program_header(m_interpreter.inode, m_interpreter.program_headers.span())))
|
|
||||||
return {};
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
ASSERT_NOT_REACHED();
|
ASSERT_NOT_REACHED();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -387,68 +374,47 @@ namespace LibELF
|
||||||
{
|
{
|
||||||
auto elf = TRY(BAN::UniqPtr<LoadableELF>::create(new_page_table));
|
auto elf = TRY(BAN::UniqPtr<LoadableELF>::create(new_page_table));
|
||||||
|
|
||||||
const auto clone_loadable_file =
|
elf->m_executable = m_executable;
|
||||||
[](const LoadableElfFile& source, LoadableElfFile& destination) -> BAN::ErrorOr<void>
|
elf->m_interpreter = m_interpreter;
|
||||||
{
|
elf->m_file_header = m_file_header;
|
||||||
if (!source.inode)
|
TRY(elf->m_program_headers.reserve(m_program_headers.size()));
|
||||||
return {};
|
for (const auto& program_header : m_program_headers)
|
||||||
|
MUST(elf->m_program_headers.emplace_back(program_header));
|
||||||
destination.inode = source.inode;
|
|
||||||
destination.file_header = source.file_header;
|
|
||||||
destination.dynamic_base = source.dynamic_base;
|
|
||||||
|
|
||||||
TRY(destination.program_headers.reserve(source.program_headers.size()));
|
|
||||||
for (const auto& program_header : source.program_headers)
|
|
||||||
MUST(destination.program_headers.emplace_back(program_header));
|
|
||||||
|
|
||||||
return {};
|
|
||||||
};
|
|
||||||
|
|
||||||
const auto map_loadable_file =
|
|
||||||
[&](BAN::Span<const ElfNativeProgramHeader> program_headers) -> BAN::ErrorOr<void>
|
|
||||||
{
|
|
||||||
for (const auto& program_header : program_headers)
|
|
||||||
{
|
|
||||||
ASSERT(program_header.p_type == PT_LOAD);
|
|
||||||
if (!(program_header.p_flags & LibELF::PF_W))
|
|
||||||
continue;
|
|
||||||
|
|
||||||
PageTable::flags_t flags = PageTable::Flags::UserSupervisor | PageTable::Flags::Present;
|
|
||||||
if (program_header.p_flags & LibELF::PF_W)
|
|
||||||
flags |= PageTable::Flags::ReadWrite;
|
|
||||||
if (program_header.p_flags & LibELF::PF_X)
|
|
||||||
flags |= PageTable::Flags::Execute;
|
|
||||||
|
|
||||||
vaddr_t start = program_header.p_vaddr & PAGE_ADDR_MASK;
|
|
||||||
size_t pages = range_page_count(program_header.p_vaddr, program_header.p_memsz);
|
|
||||||
|
|
||||||
for (size_t i = 0; i < pages; i++)
|
|
||||||
{
|
|
||||||
if (m_page_table.physical_address_of(start + i * PAGE_SIZE) == 0)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
paddr_t paddr = Heap::get().take_free_page();
|
|
||||||
if (paddr == 0)
|
|
||||||
return BAN::Error::from_errno(ENOMEM);
|
|
||||||
|
|
||||||
PageTable::with_fast_page(paddr, [&] {
|
|
||||||
memcpy(PageTable::fast_page_as_ptr(), (void*)(start + i * PAGE_SIZE), PAGE_SIZE);
|
|
||||||
});
|
|
||||||
|
|
||||||
new_page_table.map_page_at(paddr, start + i * PAGE_SIZE, flags);
|
|
||||||
elf->m_physical_page_count++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return {};
|
|
||||||
};
|
|
||||||
|
|
||||||
TRY(clone_loadable_file(m_executable, elf->m_executable));
|
|
||||||
TRY(clone_loadable_file(m_interpreter, elf->m_interpreter));
|
|
||||||
|
|
||||||
elf->reserve_address_space();
|
elf->reserve_address_space();
|
||||||
|
|
||||||
TRY(map_loadable_file(elf->m_executable.program_headers.span()));
|
for (const auto& program_header : m_program_headers)
|
||||||
TRY(map_loadable_file(elf->m_interpreter.program_headers.span()));
|
{
|
||||||
|
ASSERT(program_header.p_type == PT_LOAD);
|
||||||
|
if (!(program_header.p_flags & LibELF::PF_W))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
PageTable::flags_t flags = PageTable::Flags::UserSupervisor | PageTable::Flags::Present;
|
||||||
|
if (program_header.p_flags & LibELF::PF_W)
|
||||||
|
flags |= PageTable::Flags::ReadWrite;
|
||||||
|
if (program_header.p_flags & LibELF::PF_X)
|
||||||
|
flags |= PageTable::Flags::Execute;
|
||||||
|
|
||||||
|
vaddr_t start = program_header.p_vaddr & PAGE_ADDR_MASK;
|
||||||
|
size_t pages = range_page_count(program_header.p_vaddr, program_header.p_memsz);
|
||||||
|
|
||||||
|
for (size_t i = 0; i < pages; i++)
|
||||||
|
{
|
||||||
|
if (m_page_table.physical_address_of(start + i * PAGE_SIZE) == 0)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
paddr_t paddr = Heap::get().take_free_page();
|
||||||
|
if (paddr == 0)
|
||||||
|
return BAN::Error::from_errno(ENOMEM);
|
||||||
|
|
||||||
|
PageTable::with_fast_page(paddr, [&] {
|
||||||
|
memcpy(PageTable::fast_page_as_ptr(), (void*)(start + i * PAGE_SIZE), PAGE_SIZE);
|
||||||
|
});
|
||||||
|
|
||||||
|
new_page_table.map_page_at(paddr, start + i * PAGE_SIZE, flags);
|
||||||
|
elf->m_physical_page_count++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return elf;
|
return elf;
|
||||||
}
|
}
|
||||||
|
|
|
@ -25,10 +25,10 @@ namespace LibELF
|
||||||
static BAN::ErrorOr<BAN::UniqPtr<LoadableELF>> load_from_inode(Kernel::PageTable&, const Kernel::Credentials&, BAN::RefPtr<Kernel::Inode>);
|
static BAN::ErrorOr<BAN::UniqPtr<LoadableELF>> load_from_inode(Kernel::PageTable&, const Kernel::Credentials&, BAN::RefPtr<Kernel::Inode>);
|
||||||
~LoadableELF();
|
~LoadableELF();
|
||||||
|
|
||||||
Kernel::vaddr_t entry_point() const;
|
Kernel::vaddr_t entry_point() const { return m_file_header.e_entry; }
|
||||||
|
|
||||||
bool has_interpreter() const { return !!m_interpreter.inode; }
|
bool has_interpreter() const { return !!m_interpreter; }
|
||||||
BAN::RefPtr<Kernel::Inode> inode() { return m_executable.inode; }
|
BAN::RefPtr<Kernel::Inode> executable() { return m_executable; }
|
||||||
|
|
||||||
bool contains(Kernel::vaddr_t address) const;
|
bool contains(Kernel::vaddr_t address) const;
|
||||||
bool is_address_space_free() const;
|
bool is_address_space_free() const;
|
||||||
|
@ -44,30 +44,25 @@ namespace LibELF
|
||||||
size_t physical_page_count() const { return m_physical_page_count; }
|
size_t physical_page_count() const { return m_physical_page_count; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
struct LoadableElfFile
|
|
||||||
{
|
|
||||||
BAN::RefPtr<Kernel::Inode> inode;
|
|
||||||
ElfNativeFileHeader file_header;
|
|
||||||
BAN::Vector<ElfNativeProgramHeader> program_headers;
|
|
||||||
Kernel::vaddr_t dynamic_base;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct LoadResult
|
struct LoadResult
|
||||||
{
|
{
|
||||||
LoadableElfFile elf_file;
|
BAN::RefPtr<Kernel::Inode> inode;
|
||||||
BAN::RefPtr<Kernel::Inode> interp;
|
BAN::RefPtr<Kernel::Inode> interp;
|
||||||
|
ElfNativeFileHeader file_header;
|
||||||
|
BAN::Vector<ElfNativeProgramHeader> program_headers;
|
||||||
};
|
};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
LoadableELF(Kernel::PageTable&);
|
LoadableELF(Kernel::PageTable&);
|
||||||
BAN::ErrorOr<void> initialize(const Kernel::Credentials&, BAN::RefPtr<Kernel::Inode>);
|
BAN::ErrorOr<void> initialize(const Kernel::Credentials&, BAN::RefPtr<Kernel::Inode>);
|
||||||
|
|
||||||
bool does_executable_and_interpreter_overlap() const;
|
|
||||||
BAN::ErrorOr<LoadResult> load_elf_file(const Kernel::Credentials&, BAN::RefPtr<Kernel::Inode>) const;
|
BAN::ErrorOr<LoadResult> load_elf_file(const Kernel::Credentials&, BAN::RefPtr<Kernel::Inode>) const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
LoadableElfFile m_executable;
|
BAN::RefPtr<Kernel::Inode> m_executable;
|
||||||
LoadableElfFile m_interpreter;
|
BAN::RefPtr<Kernel::Inode> m_interpreter;
|
||||||
|
ElfNativeFileHeader m_file_header;
|
||||||
|
BAN::Vector<ElfNativeProgramHeader> m_program_headers;
|
||||||
|
|
||||||
Kernel::PageTable& m_page_table;
|
Kernel::PageTable& m_page_table;
|
||||||
size_t m_virtual_page_count { 0 };
|
size_t m_virtual_page_count { 0 };
|
||||||
size_t m_physical_page_count { 0 };
|
size_t m_physical_page_count { 0 };
|
||||||
|
|
Loading…
Reference in New Issue