mirror of
https://github.com/yuzu-emu/yuzu.git
synced 2025-01-09 08:00:05 +00:00
107 lines
2.9 KiB
C++
107 lines
2.9 KiB
C++
// SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#pragma once
|
|
|
|
#include <array>
|
|
#include <optional>
|
|
#include <type_traits>
|
|
#include "common/common_types.h"
|
|
#include "common/swap.h"
|
|
#include "core/file_sys/patch_manager.h"
|
|
#include "core/loader/loader.h"
|
|
|
|
namespace Core {
|
|
class System;
|
|
}
|
|
|
|
namespace Core::NCE {
|
|
class Patcher;
|
|
}
|
|
|
|
namespace Kernel {
|
|
class KProcess;
|
|
}
|
|
|
|
namespace Loader {
|
|
|
|
struct NSOSegmentHeader {
|
|
u32_le offset;
|
|
u32_le location;
|
|
u32_le size;
|
|
union {
|
|
u32_le alignment;
|
|
u32_le bss_size;
|
|
};
|
|
};
|
|
static_assert(sizeof(NSOSegmentHeader) == 0x10, "NsoSegmentHeader has incorrect size.");
|
|
|
|
struct NSOHeader {
|
|
using SHA256Hash = std::array<u8, 0x20>;
|
|
|
|
struct RODataRelativeExtent {
|
|
u32_le data_offset;
|
|
u32_le size;
|
|
};
|
|
|
|
u32_le magic;
|
|
u32_le version;
|
|
u32 reserved;
|
|
u32_le flags;
|
|
std::array<NSOSegmentHeader, 3> segments; // Text, RoData, Data (in that order)
|
|
std::array<u8, 0x20> build_id;
|
|
std::array<u32_le, 3> segments_compressed_size;
|
|
std::array<u8, 0x1C> padding;
|
|
RODataRelativeExtent api_info_extent;
|
|
RODataRelativeExtent dynstr_extent;
|
|
RODataRelativeExtent dynsyn_extent;
|
|
std::array<SHA256Hash, 3> segment_hashes;
|
|
|
|
bool IsSegmentCompressed(size_t segment_num) const;
|
|
};
|
|
static_assert(sizeof(NSOHeader) == 0x100, "NSOHeader has incorrect size.");
|
|
static_assert(std::is_trivially_copyable_v<NSOHeader>, "NSOHeader must be trivially copyable.");
|
|
|
|
constexpr u32 NSO_ARGUMENT_DATA_ALLOCATION_SIZE = 0x9000;
|
|
|
|
struct NSOArgumentHeader {
|
|
u32_le allocated_size;
|
|
u32_le actual_size;
|
|
INSERT_PADDING_BYTES(0x18);
|
|
};
|
|
static_assert(sizeof(NSOArgumentHeader) == 0x20, "NSOArgumentHeader has incorrect size.");
|
|
|
|
/// Loads an NSO file
|
|
class AppLoader_NSO final : public AppLoader {
|
|
public:
|
|
explicit AppLoader_NSO(FileSys::VirtualFile file_);
|
|
|
|
/**
|
|
* Identifies whether or not the given file is a form of NSO file.
|
|
*
|
|
* @param in_file The file to be identified.
|
|
*
|
|
* @return FileType::NSO if found, or FileType::Error if some other type of file.
|
|
*/
|
|
static FileType IdentifyType(const FileSys::VirtualFile& in_file);
|
|
|
|
FileType GetFileType() const override {
|
|
return IdentifyType(file);
|
|
}
|
|
|
|
static std::optional<VAddr> LoadModule(Kernel::KProcess& process, Core::System& system,
|
|
const FileSys::VfsFile& nso_file, VAddr load_base,
|
|
bool should_pass_arguments, bool load_into_process,
|
|
std::optional<FileSys::PatchManager> pm = {},
|
|
Core::NCE::Patcher* patch = nullptr);
|
|
|
|
LoadResult Load(Kernel::KProcess& process, Core::System& system) override;
|
|
|
|
ResultStatus ReadNSOModules(Modules& out_modules) override;
|
|
|
|
private:
|
|
Modules modules;
|
|
};
|
|
|
|
} // namespace Loader
|