|
|
|
@ -9,16 +9,11 @@
|
|
|
|
|
#include "common/common_types.h"
|
|
|
|
|
#include "common/file_util.h"
|
|
|
|
|
#include "common/logging/log.h"
|
|
|
|
|
#include "core/core.h"
|
|
|
|
|
#include "core/hle/kernel/kernel.h"
|
|
|
|
|
#include "core/hle/kernel/process.h"
|
|
|
|
|
#include "core/hle/kernel/vm_manager.h"
|
|
|
|
|
#include "core/loader/elf.h"
|
|
|
|
|
#include "core/memory.h"
|
|
|
|
|
|
|
|
|
|
using Kernel::CodeSet;
|
|
|
|
|
using Kernel::SharedPtr;
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// ELF Header Constants
|
|
|
|
|
|
|
|
|
@ -211,7 +206,7 @@ public:
|
|
|
|
|
u32 GetFlags() const {
|
|
|
|
|
return (u32)(header->e_flags);
|
|
|
|
|
}
|
|
|
|
|
SharedPtr<CodeSet> LoadInto(VAddr vaddr);
|
|
|
|
|
Kernel::CodeSet LoadInto(VAddr vaddr);
|
|
|
|
|
|
|
|
|
|
int GetNumSegments() const {
|
|
|
|
|
return (int)(header->e_phnum);
|
|
|
|
@ -274,7 +269,7 @@ const char* ElfReader::GetSectionName(int section) const {
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SharedPtr<CodeSet> ElfReader::LoadInto(VAddr vaddr) {
|
|
|
|
|
Kernel::CodeSet ElfReader::LoadInto(VAddr vaddr) {
|
|
|
|
|
LOG_DEBUG(Loader, "String section: {}", header->e_shstrndx);
|
|
|
|
|
|
|
|
|
|
// Should we relocate?
|
|
|
|
@ -302,8 +297,7 @@ SharedPtr<CodeSet> ElfReader::LoadInto(VAddr vaddr) {
|
|
|
|
|
std::vector<u8> program_image(total_image_size);
|
|
|
|
|
std::size_t current_image_position = 0;
|
|
|
|
|
|
|
|
|
|
auto& kernel = Core::System::GetInstance().Kernel();
|
|
|
|
|
SharedPtr<CodeSet> codeset = CodeSet::Create(kernel, "");
|
|
|
|
|
Kernel::CodeSet codeset;
|
|
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < header->e_phnum; ++i) {
|
|
|
|
|
const Elf32_Phdr* p = &segments[i];
|
|
|
|
@ -311,14 +305,14 @@ SharedPtr<CodeSet> ElfReader::LoadInto(VAddr vaddr) {
|
|
|
|
|
p->p_vaddr, p->p_filesz, p->p_memsz);
|
|
|
|
|
|
|
|
|
|
if (p->p_type == PT_LOAD) {
|
|
|
|
|
CodeSet::Segment* codeset_segment;
|
|
|
|
|
Kernel::CodeSet::Segment* codeset_segment;
|
|
|
|
|
u32 permission_flags = p->p_flags & (PF_R | PF_W | PF_X);
|
|
|
|
|
if (permission_flags == (PF_R | PF_X)) {
|
|
|
|
|
codeset_segment = &codeset->CodeSegment();
|
|
|
|
|
codeset_segment = &codeset.CodeSegment();
|
|
|
|
|
} else if (permission_flags == (PF_R)) {
|
|
|
|
|
codeset_segment = &codeset->RODataSegment();
|
|
|
|
|
codeset_segment = &codeset.RODataSegment();
|
|
|
|
|
} else if (permission_flags == (PF_R | PF_W)) {
|
|
|
|
|
codeset_segment = &codeset->DataSegment();
|
|
|
|
|
codeset_segment = &codeset.DataSegment();
|
|
|
|
|
} else {
|
|
|
|
|
LOG_ERROR(Loader, "Unexpected ELF PT_LOAD segment id {} with flags {:X}", i,
|
|
|
|
|
p->p_flags);
|
|
|
|
@ -345,8 +339,8 @@ SharedPtr<CodeSet> ElfReader::LoadInto(VAddr vaddr) {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
codeset->entrypoint = base_addr + header->e_entry;
|
|
|
|
|
codeset->memory = std::make_shared<std::vector<u8>>(std::move(program_image));
|
|
|
|
|
codeset.entrypoint = base_addr + header->e_entry;
|
|
|
|
|
codeset.memory = std::make_shared<std::vector<u8>>(std::move(program_image));
|
|
|
|
|
|
|
|
|
|
LOG_DEBUG(Loader, "Done loading.");
|
|
|
|
|
|
|
|
|
@ -397,11 +391,11 @@ ResultStatus AppLoader_ELF::Load(Kernel::Process& process) {
|
|
|
|
|
|
|
|
|
|
const VAddr base_address = process.VMManager().GetCodeRegionBaseAddress();
|
|
|
|
|
ElfReader elf_reader(&buffer[0]);
|
|
|
|
|
SharedPtr<CodeSet> codeset = elf_reader.LoadInto(base_address);
|
|
|
|
|
codeset->name = file->GetName();
|
|
|
|
|
Kernel::CodeSet codeset = elf_reader.LoadInto(base_address);
|
|
|
|
|
const VAddr entry_point = codeset.entrypoint;
|
|
|
|
|
|
|
|
|
|
process.LoadModule(codeset, codeset->entrypoint);
|
|
|
|
|
process.Run(codeset->entrypoint, 48, Memory::DEFAULT_STACK_SIZE);
|
|
|
|
|
process.LoadModule(std::move(codeset), entry_point);
|
|
|
|
|
process.Run(entry_point, 48, Memory::DEFAULT_STACK_SIZE);
|
|
|
|
|
|
|
|
|
|
is_loaded = true;
|
|
|
|
|
return ResultStatus::Success;
|
|
|
|
|