summaryrefslogtreecommitdiff
path: root/src/core/loader/kip.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/core/loader/kip.cpp')
-rw-r--r--src/core/loader/kip.cpp102
1 files changed, 102 insertions, 0 deletions
diff --git a/src/core/loader/kip.cpp b/src/core/loader/kip.cpp
new file mode 100644
index 000000000..70051c13a
--- /dev/null
+++ b/src/core/loader/kip.cpp
@@ -0,0 +1,102 @@
1// Copyright 2019 yuzu emulator team
2// Licensed under GPLv2 or any later version
3// Refer to the license.txt file included.
4
5#include "core/file_sys/kernel_executable.h"
6#include "core/file_sys/program_metadata.h"
7#include "core/gdbstub/gdbstub.h"
8#include "core/hle/kernel/code_set.h"
9#include "core/hle/kernel/process.h"
10#include "core/loader/kip.h"
11
12namespace Loader {
13
14namespace {
15constexpr u32 PageAlignSize(u32 size) {
16 return (size + Memory::PAGE_MASK) & ~Memory::PAGE_MASK;
17}
18} // Anonymous namespace
19
20AppLoader_KIP::AppLoader_KIP(FileSys::VirtualFile file_)
21 : AppLoader(std::move(file_)), kip(std::make_unique<FileSys::KIP>(file)) {}
22
23AppLoader_KIP::~AppLoader_KIP() = default;
24
25FileType AppLoader_KIP::IdentifyType(const FileSys::VirtualFile& file) {
26 u32_le magic{};
27 if (file->GetSize() < sizeof(u32) || file->ReadObject(&magic) != sizeof(u32)) {
28 return FileType::Error;
29 }
30
31 if (magic == Common::MakeMagic('K', 'I', 'P', '1')) {
32 return FileType::KIP;
33 }
34
35 return FileType::Error;
36}
37
38FileType AppLoader_KIP::GetFileType() const {
39 return (kip != nullptr && kip->GetStatus() == ResultStatus::Success) ? FileType::KIP
40 : FileType::Error;
41}
42
43AppLoader::LoadResult AppLoader_KIP::Load(Kernel::Process& process) {
44 if (is_loaded) {
45 return {ResultStatus::ErrorAlreadyLoaded, {}};
46 }
47
48 if (kip == nullptr) {
49 return {ResultStatus::ErrorNullFile, {}};
50 }
51
52 if (kip->GetStatus() != ResultStatus::Success) {
53 return {kip->GetStatus(), {}};
54 }
55
56 const auto get_kip_address_space_type = [](const auto& kip) {
57 return kip.Is64Bit()
58 ? (kip.Is39BitAddressSpace() ? FileSys::ProgramAddressSpaceType::Is39Bit
59 : FileSys::ProgramAddressSpaceType::Is36Bit)
60 : FileSys::ProgramAddressSpaceType::Is32Bit;
61 };
62
63 const auto address_space = get_kip_address_space_type(*kip);
64
65 FileSys::ProgramMetadata metadata;
66 metadata.LoadManual(kip->Is64Bit(), address_space, kip->GetMainThreadPriority(),
67 kip->GetMainThreadCpuCore(), kip->GetMainThreadStackSize(),
68 kip->GetTitleID(), 0xFFFFFFFFFFFFFFFF, kip->GetKernelCapabilities());
69
70 const VAddr base_address = process.VMManager().GetCodeRegionBaseAddress();
71 Kernel::CodeSet codeset;
72 std::vector<u8> program_image;
73
74 const auto load_segment = [&program_image](Kernel::CodeSet::Segment& segment,
75 const std::vector<u8>& data, u32 offset) {
76 segment.addr = offset;
77 segment.offset = offset;
78 segment.size = PageAlignSize(static_cast<u32>(data.size()));
79 program_image.resize(offset);
80 program_image.insert(program_image.end(), data.begin(), data.end());
81 };
82
83 load_segment(codeset.CodeSegment(), kip->GetTextSection(), kip->GetTextOffset());
84 load_segment(codeset.RODataSegment(), kip->GetRODataSection(), kip->GetRODataOffset());
85 load_segment(codeset.DataSegment(), kip->GetDataSection(), kip->GetDataOffset());
86
87 program_image.resize(PageAlignSize(kip->GetBSSOffset()) + kip->GetBSSSize());
88 codeset.DataSegment().size += kip->GetBSSSize();
89
90 GDBStub::RegisterModule(kip->GetName(), base_address, base_address + program_image.size());
91
92 codeset.memory = std::move(program_image);
93 process.LoadModule(std::move(codeset), base_address);
94
95 LOG_DEBUG(Loader, "loaded module {} @ 0x{:X}", kip->GetName(), base_address);
96
97 is_loaded = true;
98 return {ResultStatus::Success,
99 LoadParameters{kip->GetMainThreadPriority(), kip->GetMainThreadStackSize()}};
100}
101
102} // namespace Loader