Correctly handle content index sizes

This commit is contained in:
Pengfei
2021-09-09 16:06:44 +08:00
parent 5ed4b25297
commit ecdbe63c47
2 changed files with 30 additions and 5 deletions
+28 -3
View File
@@ -20,6 +20,24 @@ bool Ticket::Load(const std::vector<u8> file_data, std::size_t offset) {
return false; return false;
} }
TRY_MEMCPY(&body, file_data, offset + signature.GetSize(), sizeof(Body)); TRY_MEMCPY(&body, file_data, offset + signature.GetSize(), sizeof(Body));
// Load content index
const std::size_t content_index_offset = offset + signature.GetSize() + sizeof(Body);
struct ContentIndexHeader {
INSERT_PADDING_BYTES(4);
u32_be content_index_size;
};
ContentIndexHeader header;
TRY_MEMCPY(&header, file_data, content_index_offset, sizeof(header));
if (static_cast<u32>(header.content_index_size) > 0x10000) { // sanity limit
LOG_ERROR(Core, "Content index size too big");
return false;
}
content_index.resize(header.content_index_size);
TRY_MEMCPY(content_index.data(), file_data, content_index_offset, content_index.size());
return true; return true;
} }
@@ -35,6 +53,12 @@ bool Ticket::Save(FileUtil::IOFile& file) const {
return false; return false;
} }
// content index
if (file.WriteBytes(content_index.data(), content_index.size()) != content_index.size()) {
LOG_ERROR(Core, "Failed to save content index");
return false;
}
return true; return true;
} }
@@ -43,11 +67,12 @@ bool Ticket::ValidateSignature() const {
Common::StringFromFixedZeroTerminatedBuffer(body.issuer.data(), body.issuer.size()); Common::StringFromFixedZeroTerminatedBuffer(body.issuer.data(), body.issuer.size());
return signature.Verify(issuer, [this](CryptoPP::PK_MessageAccumulator* message) { return signature.Verify(issuer, [this](CryptoPP::PK_MessageAccumulator* message) {
message->Update(reinterpret_cast<const u8*>(&body), sizeof(body)); message->Update(reinterpret_cast<const u8*>(&body), sizeof(body));
message->Update(content_index.data(), content_index.size());
}); });
} }
std::size_t Ticket::GetSize() const { std::size_t Ticket::GetSize() const {
return signature.GetSize() + sizeof(body); return signature.GetSize() + sizeof(body) + content_index.size();
} }
constexpr std::string_view TicketIssuer = "Root-CA00000003-XS0000000c"; constexpr std::string_view TicketIssuer = "Root-CA00000003-XS0000000c";
@@ -77,9 +102,9 @@ Ticket BuildFakeTicket(u64 title_id) {
body.title_id = title_id; body.title_id = title_id;
body.common_key_index = 0x00; body.common_key_index = 0x00;
body.audit = 0x01; body.audit = 0x01;
std::memcpy(body.content_index.data(), TicketContentIndex.data(), TicketContentIndex.size()); std::memcpy(ticket.content_index.data(), TicketContentIndex.data(), TicketContentIndex.size());
// GodMode9 by default sets all remaining 0x80 bytes to 0xFF // GodMode9 by default sets all remaining 0x80 bytes to 0xFF
std::memset(body.content_index.data() + TicketContentIndex.size(), 0xFF, 0x80); std::memset(ticket.content_index.data() + TicketContentIndex.size(), 0xFF, 0x80);
return ticket; return ticket;
} }
+2 -2
View File
@@ -42,9 +42,8 @@ public:
u8 audit; u8 audit;
INSERT_PADDING_BYTES(0x42); INSERT_PADDING_BYTES(0x42);
std::array<u8, 0x40> limits; std::array<u8, 0x40> limits;
std::array<u8, 0xAC> content_index;
}; };
static_assert(sizeof(Body) == 0x210, "Ticket body structure size is wrong"); static_assert(sizeof(Body) == 0x164, "Ticket body structure size is wrong");
#pragma pack(pop) #pragma pack(pop)
bool Load(const std::vector<u8> file_data, std::size_t offset = 0); bool Load(const std::vector<u8> file_data, std::size_t offset = 0);
@@ -54,6 +53,7 @@ public:
Signature signature; Signature signature;
Body body; Body body;
std::vector<u8> content_index;
}; };
Ticket BuildFakeTicket(u64 title_id); Ticket BuildFakeTicket(u64 title_id);