2017-10-21 16:38:48 -06:00
|
|
|
// Copyright 2017 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#include <cinttypes>
|
|
|
|
#include <cryptopp/sha.h>
|
|
|
|
#include "common/alignment.h"
|
|
|
|
#include "common/file_util.h"
|
|
|
|
#include "common/logging/log.h"
|
|
|
|
#include "core/file_sys/cia_container.h"
|
|
|
|
#include "core/file_sys/file_backend.h"
|
|
|
|
#include "core/loader/loader.h"
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// FileSys namespace
|
|
|
|
|
|
|
|
namespace FileSys {
|
|
|
|
|
|
|
|
constexpr u32 CIA_SECTION_ALIGNMENT = 0x40;
|
|
|
|
|
|
|
|
Loader::ResultStatus CIAContainer::Load(const FileBackend& backend) {
|
|
|
|
std::vector<u8> header_data(sizeof(Header));
|
|
|
|
|
|
|
|
// Load the CIA Header
|
2018-09-06 16:03:28 -04:00
|
|
|
ResultVal<std::size_t> read_result = backend.Read(0, sizeof(Header), header_data.data());
|
2017-10-21 16:38:48 -06:00
|
|
|
if (read_result.Failed() || *read_result != sizeof(Header))
|
|
|
|
return Loader::ResultStatus::Error;
|
|
|
|
|
|
|
|
Loader::ResultStatus result = LoadHeader(header_data);
|
|
|
|
if (result != Loader::ResultStatus::Success)
|
|
|
|
return result;
|
|
|
|
|
2019-04-28 21:21:28 +02:00
|
|
|
// Load Ticket
|
|
|
|
std::vector<u8> ticket_data(cia_header.tik_size);
|
|
|
|
read_result = backend.Read(GetTicketOffset(), cia_header.tik_size, ticket_data.data());
|
|
|
|
if (read_result.Failed() || *read_result != cia_header.tik_size)
|
|
|
|
return Loader::ResultStatus::Error;
|
|
|
|
|
|
|
|
result = LoadTicket(ticket_data);
|
|
|
|
if (result != Loader::ResultStatus::Success)
|
|
|
|
return result;
|
|
|
|
|
2017-10-21 16:38:48 -06:00
|
|
|
// Load Title Metadata
|
|
|
|
std::vector<u8> tmd_data(cia_header.tmd_size);
|
|
|
|
read_result = backend.Read(GetTitleMetadataOffset(), cia_header.tmd_size, tmd_data.data());
|
|
|
|
if (read_result.Failed() || *read_result != cia_header.tmd_size)
|
|
|
|
return Loader::ResultStatus::Error;
|
|
|
|
|
|
|
|
result = LoadTitleMetadata(tmd_data);
|
|
|
|
if (result != Loader::ResultStatus::Success)
|
|
|
|
return result;
|
|
|
|
|
|
|
|
// Load CIA Metadata
|
|
|
|
if (cia_header.meta_size) {
|
|
|
|
std::vector<u8> meta_data(sizeof(Metadata));
|
|
|
|
read_result = backend.Read(GetMetadataOffset(), sizeof(Metadata), meta_data.data());
|
|
|
|
if (read_result.Failed() || *read_result != sizeof(Metadata))
|
|
|
|
return Loader::ResultStatus::Error;
|
|
|
|
|
|
|
|
result = LoadMetadata(meta_data);
|
|
|
|
if (result != Loader::ResultStatus::Success)
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Loader::ResultStatus::Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
Loader::ResultStatus CIAContainer::Load(const std::string& filepath) {
|
|
|
|
FileUtil::IOFile file(filepath, "rb");
|
|
|
|
if (!file.IsOpen())
|
|
|
|
return Loader::ResultStatus::Error;
|
|
|
|
|
|
|
|
// Load CIA Header
|
|
|
|
std::vector<u8> header_data(sizeof(Header));
|
|
|
|
if (file.ReadBytes(header_data.data(), sizeof(Header)) != sizeof(Header))
|
|
|
|
return Loader::ResultStatus::Error;
|
|
|
|
|
|
|
|
Loader::ResultStatus result = LoadHeader(header_data);
|
|
|
|
if (result != Loader::ResultStatus::Success)
|
|
|
|
return result;
|
|
|
|
|
2019-04-28 21:21:28 +02:00
|
|
|
// Load Ticket
|
|
|
|
std::vector<u8> ticket_data(cia_header.tik_size);
|
|
|
|
file.Seek(GetTicketOffset(), SEEK_SET);
|
|
|
|
if (file.ReadBytes(ticket_data.data(), cia_header.tik_size) != cia_header.tik_size)
|
|
|
|
return Loader::ResultStatus::Error;
|
|
|
|
|
|
|
|
result = LoadTicket(ticket_data);
|
|
|
|
if (result != Loader::ResultStatus::Success)
|
|
|
|
return result;
|
|
|
|
|
2017-10-21 16:38:48 -06:00
|
|
|
// Load Title Metadata
|
|
|
|
std::vector<u8> tmd_data(cia_header.tmd_size);
|
|
|
|
file.Seek(GetTitleMetadataOffset(), SEEK_SET);
|
2017-11-07 19:31:55 -06:00
|
|
|
if (file.ReadBytes(tmd_data.data(), cia_header.tmd_size) != cia_header.tmd_size)
|
2017-10-21 16:38:48 -06:00
|
|
|
return Loader::ResultStatus::Error;
|
|
|
|
|
|
|
|
result = LoadTitleMetadata(tmd_data);
|
|
|
|
if (result != Loader::ResultStatus::Success)
|
|
|
|
return result;
|
|
|
|
|
|
|
|
// Load CIA Metadata
|
|
|
|
if (cia_header.meta_size) {
|
|
|
|
std::vector<u8> meta_data(sizeof(Metadata));
|
|
|
|
file.Seek(GetMetadataOffset(), SEEK_SET);
|
|
|
|
if (file.ReadBytes(meta_data.data(), sizeof(Metadata)) != sizeof(Metadata))
|
|
|
|
return Loader::ResultStatus::Error;
|
|
|
|
|
|
|
|
result = LoadMetadata(meta_data);
|
|
|
|
if (result != Loader::ResultStatus::Success)
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Loader::ResultStatus::Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
Loader::ResultStatus CIAContainer::Load(const std::vector<u8>& file_data) {
|
|
|
|
Loader::ResultStatus result = LoadHeader(file_data);
|
|
|
|
if (result != Loader::ResultStatus::Success)
|
|
|
|
return result;
|
|
|
|
|
2019-04-28 21:21:28 +02:00
|
|
|
// Load Ticket
|
|
|
|
result = LoadTicket(file_data, GetTicketOffset());
|
|
|
|
if (result != Loader::ResultStatus::Success)
|
|
|
|
return result;
|
|
|
|
|
2017-10-21 16:38:48 -06:00
|
|
|
// Load Title Metadata
|
|
|
|
result = LoadTitleMetadata(file_data, GetTitleMetadataOffset());
|
|
|
|
if (result != Loader::ResultStatus::Success)
|
|
|
|
return result;
|
|
|
|
|
|
|
|
// Load CIA Metadata
|
|
|
|
if (cia_header.meta_size) {
|
|
|
|
result = LoadMetadata(file_data, GetMetadataOffset());
|
|
|
|
if (result != Loader::ResultStatus::Success)
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Loader::ResultStatus::Success;
|
|
|
|
}
|
|
|
|
|
2018-09-06 16:03:28 -04:00
|
|
|
Loader::ResultStatus CIAContainer::LoadHeader(const std::vector<u8>& header_data,
|
|
|
|
std::size_t offset) {
|
2017-10-21 16:38:48 -06:00
|
|
|
if (header_data.size() - offset < sizeof(Header))
|
|
|
|
return Loader::ResultStatus::Error;
|
|
|
|
|
|
|
|
std::memcpy(&cia_header, header_data.data(), sizeof(Header));
|
|
|
|
|
|
|
|
return Loader::ResultStatus::Success;
|
|
|
|
}
|
|
|
|
|
2018-09-04 12:28:38 -04:00
|
|
|
Loader::ResultStatus CIAContainer::LoadTicket(const std::vector<u8>& ticket_data,
|
|
|
|
std::size_t offset) {
|
|
|
|
return cia_ticket.Load(ticket_data, offset);
|
|
|
|
}
|
|
|
|
|
2017-10-21 16:38:48 -06:00
|
|
|
Loader::ResultStatus CIAContainer::LoadTitleMetadata(const std::vector<u8>& tmd_data,
|
2018-09-06 16:03:28 -04:00
|
|
|
std::size_t offset) {
|
2017-10-21 16:38:48 -06:00
|
|
|
return cia_tmd.Load(tmd_data, offset);
|
|
|
|
}
|
|
|
|
|
2018-09-06 16:03:28 -04:00
|
|
|
Loader::ResultStatus CIAContainer::LoadMetadata(const std::vector<u8>& meta_data,
|
|
|
|
std::size_t offset) {
|
2017-10-21 16:38:48 -06:00
|
|
|
if (meta_data.size() - offset < sizeof(Metadata))
|
|
|
|
return Loader::ResultStatus::Error;
|
|
|
|
|
|
|
|
std::memcpy(&cia_metadata, meta_data.data(), sizeof(Metadata));
|
|
|
|
|
|
|
|
return Loader::ResultStatus::Success;
|
|
|
|
}
|
|
|
|
|
2018-09-04 12:28:38 -04:00
|
|
|
const Ticket& CIAContainer::GetTicket() const {
|
|
|
|
return cia_ticket;
|
|
|
|
}
|
|
|
|
|
2017-10-21 16:38:48 -06:00
|
|
|
const TitleMetadata& CIAContainer::GetTitleMetadata() const {
|
|
|
|
return cia_tmd;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::array<u64, 0x30>& CIAContainer::GetDependencies() {
|
|
|
|
return cia_metadata.dependencies;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 CIAContainer::GetCoreVersion() const {
|
|
|
|
return cia_metadata.core_version;
|
|
|
|
}
|
|
|
|
|
|
|
|
u64 CIAContainer::GetCertificateOffset() const {
|
|
|
|
return Common::AlignUp(cia_header.header_size, CIA_SECTION_ALIGNMENT);
|
|
|
|
}
|
|
|
|
|
|
|
|
u64 CIAContainer::GetTicketOffset() const {
|
|
|
|
return Common::AlignUp(GetCertificateOffset() + cia_header.cert_size, CIA_SECTION_ALIGNMENT);
|
|
|
|
}
|
|
|
|
|
|
|
|
u64 CIAContainer::GetTitleMetadataOffset() const {
|
|
|
|
return Common::AlignUp(GetTicketOffset() + cia_header.tik_size, CIA_SECTION_ALIGNMENT);
|
|
|
|
}
|
|
|
|
|
|
|
|
u64 CIAContainer::GetMetadataOffset() const {
|
|
|
|
u64 tmd_end_offset = GetContentOffset();
|
|
|
|
|
|
|
|
// Meta exists after all content in the CIA
|
|
|
|
u64 offset = Common::AlignUp(tmd_end_offset + cia_header.content_size, CIA_SECTION_ALIGNMENT);
|
|
|
|
|
|
|
|
return offset;
|
|
|
|
}
|
|
|
|
|
2020-06-20 14:09:56 -04:00
|
|
|
u64 CIAContainer::GetContentOffset(std::size_t index) const {
|
2017-10-21 16:38:48 -06:00
|
|
|
u64 offset =
|
|
|
|
Common::AlignUp(GetTitleMetadataOffset() + cia_header.tmd_size, CIA_SECTION_ALIGNMENT);
|
2020-06-20 14:09:56 -04:00
|
|
|
for (std::size_t i = 0; i < index; i++) {
|
2017-10-21 16:38:48 -06:00
|
|
|
offset += GetContentSize(i);
|
|
|
|
}
|
|
|
|
return offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 CIAContainer::GetCertificateSize() const {
|
|
|
|
return cia_header.cert_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 CIAContainer::GetTicketSize() const {
|
|
|
|
return cia_header.tik_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 CIAContainer::GetTitleMetadataSize() const {
|
|
|
|
return cia_header.tmd_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 CIAContainer::GetMetadataSize() const {
|
|
|
|
return cia_header.meta_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
u64 CIAContainer::GetTotalContentSize() const {
|
|
|
|
return cia_header.content_size;
|
|
|
|
}
|
|
|
|
|
2020-06-20 14:09:56 -04:00
|
|
|
u64 CIAContainer::GetContentSize(std::size_t index) const {
|
2017-10-21 16:38:48 -06:00
|
|
|
// If the content doesn't exist in the CIA, it doesn't have a size.
|
2020-06-20 14:09:56 -04:00
|
|
|
if (!cia_header.IsContentPresent(index)) {
|
2017-10-21 16:38:48 -06:00
|
|
|
return 0;
|
2020-06-20 14:09:56 -04:00
|
|
|
}
|
2017-10-21 16:38:48 -06:00
|
|
|
|
|
|
|
return cia_tmd.GetContentSizeByIndex(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CIAContainer::Print() const {
|
2018-06-29 14:18:07 +03:00
|
|
|
LOG_DEBUG(Service_FS, "Type: {}", static_cast<u32>(cia_header.type));
|
|
|
|
LOG_DEBUG(Service_FS, "Version: {}\n", static_cast<u32>(cia_header.version));
|
|
|
|
|
|
|
|
LOG_DEBUG(Service_FS, "Certificate Size: 0x{:08x} bytes", GetCertificateSize());
|
|
|
|
LOG_DEBUG(Service_FS, "Ticket Size: 0x{:08x} bytes", GetTicketSize());
|
|
|
|
LOG_DEBUG(Service_FS, "TMD Size: 0x{:08x} bytes", GetTitleMetadataSize());
|
|
|
|
LOG_DEBUG(Service_FS, "Meta Size: 0x{:08x} bytes", GetMetadataSize());
|
|
|
|
LOG_DEBUG(Service_FS, "Content Size: 0x{:08x} bytes\n", GetTotalContentSize());
|
|
|
|
|
|
|
|
LOG_DEBUG(Service_FS, "Certificate Offset: 0x{:08x} bytes", GetCertificateOffset());
|
|
|
|
LOG_DEBUG(Service_FS, "Ticket Offset: 0x{:08x} bytes", GetTicketOffset());
|
|
|
|
LOG_DEBUG(Service_FS, "TMD Offset: 0x{:08x} bytes", GetTitleMetadataOffset());
|
|
|
|
LOG_DEBUG(Service_FS, "Meta Offset: 0x{:08x} bytes", GetMetadataOffset());
|
2017-10-21 16:38:48 -06:00
|
|
|
for (u16 i = 0; i < cia_tmd.GetContentCount(); i++) {
|
2018-06-29 14:18:07 +03:00
|
|
|
LOG_DEBUG(Service_FS, "Content {:x} Offset: 0x{:08x} bytes", i, GetContentOffset(i));
|
2017-10-21 16:38:48 -06:00
|
|
|
}
|
|
|
|
}
|
2018-03-09 10:54:43 -07:00
|
|
|
} // namespace FileSys
|