2018-01-13 21:22:39 +00:00
|
|
|
// Copyright 2018 yuzu emulator team
|
2017-09-24 16:08:31 +01:00
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2019-03-22 17:04:41 +00:00
|
|
|
#include <array>
|
2018-10-15 02:41:58 +01:00
|
|
|
#include <optional>
|
2019-03-22 17:04:41 +00:00
|
|
|
#include <type_traits>
|
2017-09-24 16:08:31 +01:00
|
|
|
#include "common/common_types.h"
|
2019-03-20 00:45:55 +00:00
|
|
|
#include "common/swap.h"
|
2018-09-30 03:16:09 +01:00
|
|
|
#include "core/file_sys/patch_manager.h"
|
2017-09-24 16:08:31 +01:00
|
|
|
#include "core/loader/loader.h"
|
|
|
|
|
2018-12-03 03:24:43 +00:00
|
|
|
namespace Kernel {
|
|
|
|
class Process;
|
|
|
|
}
|
|
|
|
|
2017-09-24 16:08:31 +01:00
|
|
|
namespace Loader {
|
|
|
|
|
2019-03-22 17:04:41 +00:00
|
|
|
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.");
|
|
|
|
|
2018-10-05 18:52:07 +01:00
|
|
|
constexpr u64 NSO_ARGUMENT_DATA_ALLOCATION_SIZE = 0x9000;
|
|
|
|
|
2018-09-30 19:04:48 +01:00
|
|
|
struct NSOArgumentHeader {
|
|
|
|
u32_le allocated_size;
|
|
|
|
u32_le actual_size;
|
|
|
|
INSERT_PADDING_BYTES(0x18);
|
|
|
|
};
|
|
|
|
static_assert(sizeof(NSOArgumentHeader) == 0x20, "NSOArgumentHeader has incorrect size.");
|
|
|
|
|
2017-09-24 16:08:31 +01:00
|
|
|
/// Loads an NSO file
|
2019-03-20 00:45:55 +00:00
|
|
|
class AppLoader_NSO final : public AppLoader {
|
2017-09-24 16:08:31 +01:00
|
|
|
public:
|
2018-07-19 02:07:11 +01:00
|
|
|
explicit AppLoader_NSO(FileSys::VirtualFile file);
|
2017-09-24 16:08:31 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the type of the file
|
2018-07-19 02:07:11 +01:00
|
|
|
* @param file std::shared_ptr<VfsFile> open file
|
2017-09-24 16:08:31 +01:00
|
|
|
* @return FileType found, or FileType::Error if this loader doesn't know it
|
|
|
|
*/
|
2018-07-19 02:07:11 +01:00
|
|
|
static FileType IdentifyType(const FileSys::VirtualFile& file);
|
2017-09-24 16:08:31 +01:00
|
|
|
|
2018-12-05 22:42:41 +00:00
|
|
|
FileType GetFileType() const override {
|
2018-07-19 02:07:11 +01:00
|
|
|
return IdentifyType(file);
|
2017-09-24 16:08:31 +01:00
|
|
|
}
|
|
|
|
|
2018-12-03 03:24:43 +00:00
|
|
|
static std::optional<VAddr> LoadModule(Kernel::Process& process, const FileSys::VfsFile& file,
|
|
|
|
VAddr load_base, bool should_pass_arguments,
|
2018-10-15 02:41:58 +01:00
|
|
|
std::optional<FileSys::PatchManager> pm = {});
|
2018-01-20 19:20:04 +00:00
|
|
|
|
2019-04-09 22:03:04 +01:00
|
|
|
LoadResult Load(Kernel::Process& process) override;
|
2019-05-26 16:40:41 +01:00
|
|
|
|
|
|
|
ResultStatus ReadNSOModules(Modules& modules) override;
|
|
|
|
|
|
|
|
private:
|
|
|
|
Modules modules;
|
2017-09-24 16:08:31 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace Loader
|