2018-06-21 10:16:23 -05:00
|
|
|
// Copyright 2018 yuzu emulator team
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2018-07-19 23:10:21 -05:00
|
|
|
#include <utility>
|
2018-06-21 10:16:23 -05:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include "common/file_util.h"
|
|
|
|
#include "common/logging/log.h"
|
2018-07-12 22:22:59 -05:00
|
|
|
#include "common/string_util.h"
|
2018-07-07 22:24:51 -05:00
|
|
|
#include "common/swap.h"
|
2018-06-21 10:16:23 -05:00
|
|
|
#include "core/core.h"
|
2018-07-18 20:07:11 -05:00
|
|
|
#include "core/file_sys/content_archive.h"
|
2018-06-21 10:16:23 -05:00
|
|
|
#include "core/file_sys/program_metadata.h"
|
2018-07-12 22:22:59 -05:00
|
|
|
#include "core/gdbstub/gdbstub.h"
|
2018-06-21 10:16:23 -05:00
|
|
|
#include "core/hle/kernel/process.h"
|
|
|
|
#include "core/hle/kernel/resource_limit.h"
|
|
|
|
#include "core/hle/service/filesystem/filesystem.h"
|
|
|
|
#include "core/loader/nca.h"
|
|
|
|
#include "core/loader/nso.h"
|
|
|
|
#include "core/memory.h"
|
|
|
|
|
|
|
|
namespace Loader {
|
|
|
|
|
2018-07-19 23:10:21 -05:00
|
|
|
AppLoader_NCA::AppLoader_NCA(FileSys::VirtualFile file) : AppLoader(std::move(file)) {}
|
2018-06-21 10:16:23 -05:00
|
|
|
|
2018-07-18 20:07:11 -05:00
|
|
|
FileType AppLoader_NCA::IdentifyType(const FileSys::VirtualFile& file) {
|
2018-07-07 22:24:51 -05:00
|
|
|
// TODO(DarkLordZach): Assuming everything is decrypted. Add crypto support.
|
2018-07-18 20:07:11 -05:00
|
|
|
FileSys::NCAHeader header{};
|
|
|
|
if (sizeof(FileSys::NCAHeader) != file->ReadObject(&header))
|
|
|
|
return FileType::Error;
|
2018-07-07 22:24:51 -05:00
|
|
|
|
2018-07-18 20:07:11 -05:00
|
|
|
if (IsValidNCA(header) && header.content_type == FileSys::NCAContentType::Program)
|
2018-06-21 10:16:23 -05:00
|
|
|
return FileType::NCA;
|
|
|
|
|
|
|
|
return FileType::Error;
|
|
|
|
}
|
|
|
|
|
|
|
|
ResultStatus AppLoader_NCA::Load(Kernel::SharedPtr<Kernel::Process>& process) {
|
|
|
|
if (is_loaded) {
|
|
|
|
return ResultStatus::ErrorAlreadyLoaded;
|
|
|
|
}
|
|
|
|
|
2018-07-18 20:07:11 -05:00
|
|
|
nca = std::make_unique<FileSys::NCA>(file);
|
|
|
|
ResultStatus result = nca->GetStatus();
|
2018-06-21 10:16:23 -05:00
|
|
|
if (result != ResultStatus::Success) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-07-18 20:07:11 -05:00
|
|
|
if (nca->GetType() != FileSys::NCAContentType::Program)
|
|
|
|
return ResultStatus::ErrorInvalidFormat;
|
|
|
|
|
|
|
|
auto exefs = nca->GetExeFS();
|
|
|
|
|
|
|
|
if (exefs == nullptr)
|
|
|
|
return ResultStatus::ErrorInvalidFormat;
|
|
|
|
|
|
|
|
result = metadata.Load(exefs->GetFile("main.npdm"));
|
2018-06-21 10:16:23 -05:00
|
|
|
if (result != ResultStatus::Success) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
metadata.Print();
|
|
|
|
|
|
|
|
const FileSys::ProgramAddressSpaceType arch_bits{metadata.GetAddressSpaceType()};
|
|
|
|
if (arch_bits == FileSys::ProgramAddressSpaceType::Is32Bit) {
|
|
|
|
return ResultStatus::ErrorUnsupportedArch;
|
|
|
|
}
|
|
|
|
|
|
|
|
VAddr next_load_addr{Memory::PROCESS_IMAGE_VADDR};
|
|
|
|
for (const auto& module : {"rtld", "main", "subsdk0", "subsdk1", "subsdk2", "subsdk3",
|
|
|
|
"subsdk4", "subsdk5", "subsdk6", "subsdk7", "sdk"}) {
|
|
|
|
const VAddr load_addr = next_load_addr;
|
2018-07-18 20:07:11 -05:00
|
|
|
|
|
|
|
next_load_addr = AppLoader_NSO::LoadModule(exefs->GetFile(module), load_addr);
|
2018-06-21 10:16:23 -05:00
|
|
|
if (next_load_addr) {
|
2018-07-02 11:13:26 -05:00
|
|
|
LOG_DEBUG(Loader, "loaded module {} @ 0x{:X}", module, load_addr);
|
2018-07-12 22:22:59 -05:00
|
|
|
// Register module with GDBStub
|
|
|
|
GDBStub::RegisterModule(module, load_addr, next_load_addr - 1, false);
|
2018-06-21 10:16:23 -05:00
|
|
|
} else {
|
|
|
|
next_load_addr = load_addr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
process->program_id = metadata.GetTitleID();
|
|
|
|
process->svc_access_mask.set();
|
|
|
|
process->address_mappings = default_address_mappings;
|
|
|
|
process->resource_limit =
|
|
|
|
Kernel::ResourceLimit::GetForCategory(Kernel::ResourceLimitCategory::APPLICATION);
|
|
|
|
process->Run(Memory::PROCESS_IMAGE_VADDR, metadata.GetMainThreadPriority(),
|
|
|
|
metadata.GetMainThreadStackSize());
|
|
|
|
|
2018-07-18 20:07:11 -05:00
|
|
|
if (nca->GetRomFS() != nullptr && nca->GetRomFS()->GetSize() > 0)
|
2018-07-17 14:42:15 -05:00
|
|
|
Service::FileSystem::RegisterRomFS(std::make_unique<FileSys::RomFSFactory>(*this));
|
2018-07-07 22:24:51 -05:00
|
|
|
|
2018-06-21 10:16:23 -05:00
|
|
|
is_loaded = true;
|
2018-07-18 20:07:11 -05:00
|
|
|
|
2018-07-07 22:24:51 -05:00
|
|
|
return ResultStatus::Success;
|
|
|
|
}
|
|
|
|
|
2018-07-18 20:07:11 -05:00
|
|
|
ResultStatus AppLoader_NCA::ReadRomFS(FileSys::VirtualFile& dir) {
|
|
|
|
if (nca == nullptr || nca->GetRomFS() == nullptr || nca->GetRomFS()->GetSize() == 0)
|
2018-07-07 22:24:51 -05:00
|
|
|
return ResultStatus::ErrorNotUsed;
|
2018-07-18 20:07:11 -05:00
|
|
|
dir = nca->GetRomFS();
|
2018-06-21 10:16:23 -05:00
|
|
|
return ResultStatus::Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
AppLoader_NCA::~AppLoader_NCA() = default;
|
|
|
|
|
|
|
|
} // namespace Loader
|