summaryrefslogtreecommitdiff
path: root/src/core/hle
diff options
context:
space:
mode:
authorVolcaEM <63682805+VolcaEM@users.noreply.github.com>2020-06-14 19:28:39 +0200
committerGitHub <noreply@github.com>2020-06-14 19:28:39 +0200
commit151a3fe7b327c8adad2bc103029457c353b261c0 (patch)
tree624fd7ee4545d36c4bfecec4ce8922b880b0063e /src/core/hle
parentdfd1badc125ca5a4a4fa4587b81f2e51bebb2a07 (diff)
Attempt to fix crashes in SSBU and refactor IsValidNRO
Diffstat (limited to 'src/core/hle')
-rw-r--r--src/core/hle/service/ldr/ldr.cpp95
1 files changed, 59 insertions, 36 deletions
diff --git a/src/core/hle/service/ldr/ldr.cpp b/src/core/hle/service/ldr/ldr.cpp
index 69caf3aae..5b372b7db 100644
--- a/src/core/hle/service/ldr/ldr.cpp
+++ b/src/core/hle/service/ldr/ldr.cpp
@@ -39,22 +39,27 @@ constexpr ResultCode ERROR_NOT_INITIALIZED{ErrorModule::Loader, 87};
constexpr std::size_t MAXIMUM_LOADED_RO{0x40};
constexpr std::size_t MAXIMUM_MAP_RETRIES{0x200};
-struct Certification {
+constexpr std::size_t TEXT_INDEX{0};
+constexpr std::size_t RO_INDEX{1};
+constexpr std::size_t DATA_INDEX{2};
+
+struct NRRCertification {
u64_le application_id_mask;
u64_le application_id_pattern;
std::array<u8, 0x10> reserved;
std::array<u8, 0x100> public_key; // Also known as modulus
std::array<u8, 0x100> signature;
};
-static_assert(sizeof(Certification) == 0x220, "Certification has invalid size!");
+static_assert(sizeof(NRRCertification) == 0x220, "Certification has invalid size!");
using SHA256Hash = std::array<u8, 0x20>;
+#pragma pack(1)
struct NRRHeader {
u32_le magic;
u32_le certification_signature_key_generation; // 9.0.0+
u64_le reserved;
- Certification certification;
+ NRRCertification certification;
std::array<u8, 0x100> signature;
u64_le application_id;
u32_le size;
@@ -63,21 +68,19 @@ struct NRRHeader {
u32_le hash_offset;
u32_le hash_count;
u64_le reserved_3;
-
- // Must be dynamically allocated because, according to
- // SwitchBrew, its size is (0x20 * hash_count) and
- // it's impossible to determine the value of hash_count
- // (SwitchBrew calls it "NumHash") before runtime,
- // therefore it's not possible to calculate a SHA-256
- std::vector<SHA256Hash> NroHashList;
};
+#pragma pack()
+static_assert(sizeof(NRRHeader) == 0x350, "NRRHeader has invalid size!");
+#pragma pack(1)
struct SegmentHeader {
u32_le memory_offset;
u32_le memory_size;
};
+#pragma pack()
static_assert(sizeof(SegmentHeader) == 0x8, "SegmentHeader has invalid size!");
+#pragma pack(1)
struct NROHeader {
// Switchbrew calls this "Start" (0x10)
u32_le unused;
@@ -99,8 +102,10 @@ struct NROHeader {
// .apiInfo, .dynstr, .dynsym
std::array<SegmentHeader, 3> segment_headers_2;
};
+#pragma pack()
static_assert(sizeof(NROHeader) == 0x80, "NROHeader has invalid size.");
+#pragma pack(1)
struct NROInfo {
SHA256Hash hash{};
VAddr nro_address{};
@@ -112,6 +117,8 @@ struct NROInfo {
std::size_t data_size{};
VAddr src_addr{};
};
+#pragma pack()
+static_assert(sizeof(NROInfo) == 0x60, "NROInfo has invalid size.");
class DebugMonitor final : public ServiceFramework<DebugMonitor> {
public:
@@ -369,10 +376,10 @@ public:
ResultCode LoadNro(Kernel::Process* process, const NROHeader& nro_header, VAddr nro_addr,
VAddr start) const {
- const VAddr text_start{start + nro_header.segment_headers[0].memory_offset};
- const VAddr ro_start{start + nro_header.segment_headers[1].memory_offset};
- const VAddr data_start{start + nro_header.segment_headers[2].memory_offset};
- const VAddr bss_start{data_start + nro_header.segment_headers[2].memory_size};
+ const VAddr text_start{start + nro_header.segment_headers[TEXT_INDEX].memory_offset};
+ const VAddr ro_start{start + nro_header.segment_headers[RO_INDEX].memory_offset};
+ const VAddr data_start{start + nro_header.segment_headers[DATA_INDEX].memory_offset};
+ const VAddr bss_start{data_start + nro_header.segment_headers[DATA_INDEX].memory_size};
const VAddr bss_end_addr{
Common::AlignUp(bss_start + nro_header.bss_size, Kernel::Memory::PageSize)};
@@ -381,12 +388,12 @@ public:
system.Memory().ReadBlock(src_addr, source_data.data(), source_data.size());
system.Memory().WriteBlock(dst_addr, source_data.data(), source_data.size());
}};
- CopyCode(nro_addr + nro_header.segment_headers[0].memory_offset, text_start,
- nro_header.segment_headers[0].memory_size);
- CopyCode(nro_addr + nro_header.segment_headers[1].memory_offset, ro_start,
- nro_header.segment_headers[1].memory_size);
- CopyCode(nro_addr + nro_header.segment_headers[2].memory_offset, data_start,
- nro_header.segment_headers[2].memory_size);
+ CopyCode(nro_addr + nro_header.segment_headers[TEXT_INDEX].memory_offset, text_start,
+ nro_header.segment_headers[TEXT_INDEX].memory_size);
+ CopyCode(nro_addr + nro_header.segment_headers[RO_INDEX].memory_offset, ro_start,
+ nro_header.segment_headers[RO_INDEX].memory_size);
+ CopyCode(nro_addr + nro_header.segment_headers[DATA_INDEX].memory_offset, data_start,
+ nro_header.segment_headers[DATA_INDEX].memory_size);
CASCADE_CODE(process->PageTable().SetCodeMemoryPermission(
text_start, ro_start - text_start, Kernel::Memory::MemoryPermission::ReadAndExecute));
@@ -510,9 +517,9 @@ public:
// Track the loaded NRO
nro.insert_or_assign(*map_result,
NROInfo{hash, *map_result, nro_size, bss_address, bss_size,
- header.segment_headers[0].memory_size,
- header.segment_headers[1].memory_size,
- header.segment_headers[2].memory_size, nro_address});
+ header.segment_headers[TEXT_INDEX].memory_size,
+ header.segment_headers[RO_INDEX].memory_size,
+ header.segment_headers[DATA_INDEX].memory_size, nro_address});
// Invalidate JIT caches for the newly mapped process code
system.InvalidateCpuInstructionCaches();
@@ -608,19 +615,35 @@ private:
}
static bool IsValidNRO(const NROHeader& header, u64 nro_size, u64 bss_size) {
- return header.magic == Common::MakeMagic('N', 'R', 'O', '0') &&
- header.nro_size == nro_size && header.bss_size == bss_size &&
- header.segment_headers[1].memory_offset ==
- header.segment_headers[0].memory_offset +
- header.segment_headers[0].memory_size &&
- header.segment_headers[2].memory_offset ==
- header.segment_headers[1].memory_offset +
- header.segment_headers[1].memory_size &&
- nro_size == header.segment_headers[2].memory_offset +
- header.segment_headers[2].memory_size &&
- Common::Is4KBAligned(header.segment_headers[0].memory_size) &&
- Common::Is4KBAligned(header.segment_headers[1].memory_size) &&
- Common::Is4KBAligned(header.segment_headers[2].memory_size);
+
+ const bool valid_magic = header.magic == Common::MakeMagic('N', 'R', 'O', '0');
+
+ const bool valid_nro_size = header.nro_size == nro_size;
+
+ const bool valid_bss_size = header.bss_size == bss_size;
+
+ const bool valid_ro_offset = header.segment_headers[RO_INDEX].memory_offset ==
+ header.segment_headers[TEXT_INDEX].memory_offset +
+ header.segment_headers[TEXT_INDEX].memory_size;
+
+ const bool valid_rw_offset = header.segment_headers[DATA_INDEX].memory_offset ==
+ header.segment_headers[RO_INDEX].memory_offset +
+ header.segment_headers[RO_INDEX].memory_size;
+
+ const bool valid_nro_calculated_size =
+ nro_size == header.segment_headers[DATA_INDEX].memory_offset +
+ header.segment_headers[DATA_INDEX].memory_size;
+
+ const bool text_aligned =
+ Common::Is4KBAligned(header.segment_headers[TEXT_INDEX].memory_size);
+
+ const bool ro_aligned = Common::Is4KBAligned(header.segment_headers[RO_INDEX].memory_size);
+
+ const bool rw_aligned =
+ Common::Is4KBAligned(header.segment_headers[DATA_INDEX].memory_size);
+
+ return valid_magic && valid_nro_size && valid_bss_size && valid_ro_offset &&
+ valid_rw_offset && valid_nro_calculated_size && text_aligned && ro_aligned && rw_aligned;
}
Core::System& system;
};