summaryrefslogtreecommitdiff
path: root/src/core/loader/ncch.cpp
diff options
context:
space:
mode:
authorGravatar Yuri Kunde Schlesner2015-07-11 20:07:49 -0700
committerGravatar Yuri Kunde Schlesner2015-07-11 20:07:49 -0700
commit4e900d56f3dd2b5c9871b523689322028123d891 (patch)
tree2b233263cff7c001506f660373e2364c8e702637 /src/core/loader/ncch.cpp
parentMerge pull request #914 from yuriks/bitfield-mask (diff)
parentCore: Properly configure address space when loading a binary (diff)
downloadyuzu-4e900d56f3dd2b5c9871b523689322028123d891.tar.gz
yuzu-4e900d56f3dd2b5c9871b523689322028123d891.tar.xz
yuzu-4e900d56f3dd2b5c9871b523689322028123d891.zip
Merge pull request #912 from yuriks/process-loading
Core: Properly configure address space during binary loading
Diffstat (limited to 'src/core/loader/ncch.cpp')
-rw-r--r--src/core/loader/ncch.cpp32
1 files changed, 28 insertions, 4 deletions
diff --git a/src/core/loader/ncch.cpp b/src/core/loader/ncch.cpp
index 2b26b31cf..87603d198 100644
--- a/src/core/loader/ncch.cpp
+++ b/src/core/loader/ncch.cpp
@@ -118,6 +118,9 @@ FileType AppLoader_NCCH::IdentifyType(FileUtil::IOFile& file) {
118} 118}
119 119
120ResultStatus AppLoader_NCCH::LoadExec() const { 120ResultStatus AppLoader_NCCH::LoadExec() const {
121 using Kernel::SharedPtr;
122 using Kernel::CodeSet;
123
121 if (!is_loaded) 124 if (!is_loaded)
122 return ResultStatus::ErrorNotLoaded; 125 return ResultStatus::ErrorNotLoaded;
123 126
@@ -126,7 +129,30 @@ ResultStatus AppLoader_NCCH::LoadExec() const {
126 std::string process_name = Common::StringFromFixedZeroTerminatedBuffer( 129 std::string process_name = Common::StringFromFixedZeroTerminatedBuffer(
127 (const char*)exheader_header.codeset_info.name, 8); 130 (const char*)exheader_header.codeset_info.name, 8);
128 u64 program_id = *reinterpret_cast<u64_le const*>(&ncch_header.program_id[0]); 131 u64 program_id = *reinterpret_cast<u64_le const*>(&ncch_header.program_id[0]);
129 Kernel::g_current_process = Kernel::Process::Create(process_name, program_id); 132
133 SharedPtr<CodeSet> codeset = CodeSet::Create(process_name, program_id);
134
135 codeset->code.offset = 0;
136 codeset->code.addr = exheader_header.codeset_info.text.address;
137 codeset->code.size = exheader_header.codeset_info.text.num_max_pages * Memory::PAGE_SIZE;
138
139 codeset->rodata.offset = codeset->code.offset + codeset->code.size;
140 codeset->rodata.addr = exheader_header.codeset_info.ro.address;
141 codeset->rodata.size = exheader_header.codeset_info.ro.num_max_pages * Memory::PAGE_SIZE;
142
143 // TODO(yuriks): Not sure if the bss size is added to the page-aligned .data size or just
144 // to the regular size. Playing it safe for now.
145 u32 bss_page_size = (exheader_header.codeset_info.bss_size + 0xFFF) & ~0xFFF;
146 code.resize(code.size() + bss_page_size, 0);
147
148 codeset->data.offset = codeset->rodata.offset + codeset->rodata.size;
149 codeset->data.addr = exheader_header.codeset_info.data.address;
150 codeset->data.size = exheader_header.codeset_info.data.num_max_pages * Memory::PAGE_SIZE + bss_page_size;
151
152 codeset->entrypoint = codeset->code.addr;
153 codeset->memory = std::make_shared<std::vector<u8>>(std::move(code));
154
155 Kernel::g_current_process = Kernel::Process::Create(std::move(codeset));
130 156
131 // Attach a resource limit to the process based on the resource limit category 157 // Attach a resource limit to the process based on the resource limit category
132 Kernel::g_current_process->resource_limit = Kernel::ResourceLimit::GetForCategory( 158 Kernel::g_current_process->resource_limit = Kernel::ResourceLimit::GetForCategory(
@@ -137,11 +163,9 @@ ResultStatus AppLoader_NCCH::LoadExec() const {
137 std::copy_n(exheader_header.arm11_kernel_caps.descriptors, kernel_caps.size(), begin(kernel_caps)); 163 std::copy_n(exheader_header.arm11_kernel_caps.descriptors, kernel_caps.size(), begin(kernel_caps));
138 Kernel::g_current_process->ParseKernelCaps(kernel_caps.data(), kernel_caps.size()); 164 Kernel::g_current_process->ParseKernelCaps(kernel_caps.data(), kernel_caps.size());
139 165
140 Memory::WriteBlock(entry_point, &code[0], code.size());
141
142 s32 priority = exheader_header.arm11_system_local_caps.priority; 166 s32 priority = exheader_header.arm11_system_local_caps.priority;
143 u32 stack_size = exheader_header.codeset_info.stack_size; 167 u32 stack_size = exheader_header.codeset_info.stack_size;
144 Kernel::g_current_process->Run(entry_point, priority, stack_size); 168 Kernel::g_current_process->Run(priority, stack_size);
145 return ResultStatus::Success; 169 return ResultStatus::Success;
146 } 170 }
147 return ResultStatus::Error; 171 return ResultStatus::Error;