/* * Copyright (c) Atmosphère-NX * * This program is free software; you can redistribute it and/or modify it * under the terms and conditions of the GNU General Public License, * version 2, as published by the Free Software Foundation. * * This program is distributed in the hope it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #pragma once #include #include #include #include #include namespace ams::ncm { enum class UpdateType : u8 { ApplyAsDelta = 0, Overwrite = 1, Create = 2, }; struct FragmentIndicator { u16 content_info_index; u16 fragment_index; }; struct FragmentSet { ContentId source_content_id; ContentId destination_content_id; u32 source_size_low; u16 source_size_high; u16 destination_size_high; u32 destination_size_low; u16 fragment_count; ContentType target_content_type; UpdateType update_type; u8 reserved[4]; constexpr s64 GetSourceSize() const { return (static_cast(this->source_size_high) << 32) + this->source_size_low; } constexpr s64 GetDestinationSize() const { return (static_cast(this->destination_size_high) << 32) + this->destination_size_low; } constexpr void SetSourceSize(s64 size) { this->source_size_low = size & 0xFFFFFFFFll; this->source_size_high = static_cast(size >> 32); } constexpr void SetDestinationSize(s64 size) { this->destination_size_low = size & 0xFFFFFFFFll; this->destination_size_high = static_cast(size >> 32); } }; struct SystemUpdateMetaExtendedDataHeader { u32 version; u32 firmware_variation_count; }; struct DeltaMetaExtendedDataHeader { PatchId source_id; PatchId destination_id; u32 source_version; u32 destination_version; u16 fragment_set_count; u8 reserved[6]; }; struct PatchMetaExtendedDataHeader { u32 history_count; u32 delta_history_count; u32 delta_count; u32 fragment_set_count; u32 history_content_total_count; u32 delta_content_total_count; u8 reserved[4]; }; struct PatchHistoryHeader { ContentMetaKey key; Digest digest; u16 content_count; u8 reserved[2]; }; struct PatchDeltaHistory { PatchId source_id; PatchId destination_id; u32 source_version; u32 destination_version; u64 download_size; u8 reserved[4]; }; struct PatchDeltaHeader { DeltaMetaExtendedDataHeader delta; u16 content_count; u8 reserved[4]; }; template class PatchMetaExtendedDataReaderWriterBase { private: MemberTypePointer m_data; const size_t m_size; public: PatchMetaExtendedDataReaderWriterBase(MemberTypePointer d, size_t sz) : m_data(d), m_size(sz) { /* ... */ } protected: s32 CountFragmentSet(s32 delta_index) const { auto delta_header = this->GetPatchDeltaHeader(0); s32 count = 0; for (s32 i = 0; i < delta_index; i++) { count += delta_header[i].delta.fragment_set_count; } return count; } s32 CountHistoryContent(s32 history_index) const { auto history_header = this->GetPatchHistoryHeader(0); s32 count = 0; for (s32 i = 0; i < history_index; i++) { count += history_header[i].content_count; } return count; } s32 CountDeltaContent(s32 delta_index) const { auto delta_header = this->GetPatchDeltaHeader(0); s32 count = 0; for (s32 i = 0; i < delta_index; i++) { count += delta_header[i].content_count; } return count; } s32 CountFragment(s32 index) const { auto fragment_set = this->GetFragmentSet(0, 0); s32 count = 0; for (s32 i = 0; i < index; i++) { count += fragment_set[i].fragment_count; } return count; } DataTypePointer GetHeaderAddress() const { return reinterpret_cast(m_data); } DataTypePointer GetPatchHistoryHeaderAddress(s32 index) const { auto header = this->GetHeader(); AMS_ABORT_UNLESS(static_cast(index) < header->history_count); return this->GetHeaderAddress() + sizeof(PatchMetaExtendedDataHeader) + sizeof(PatchHistoryHeader) * index; } DataTypePointer GetPatchDeltaHistoryAddress(s32 index) const { auto header = this->GetHeader(); AMS_ABORT_UNLESS(static_cast(index) < header->delta_history_count); return this->GetHeaderAddress() + sizeof(PatchMetaExtendedDataHeader) + sizeof(PatchHistoryHeader) * header->history_count + sizeof(PatchDeltaHistory) * index; } DataTypePointer GetPatchDeltaHeaderAddress(s32 index) const { auto header = this->GetHeader(); AMS_ABORT_UNLESS(static_cast(index) < header->delta_count); return this->GetHeaderAddress() + sizeof(PatchMetaExtendedDataHeader) + sizeof(PatchHistoryHeader) * header->history_count + sizeof(PatchDeltaHistory) * header->delta_history_count + sizeof(PatchDeltaHeader) * index; } DataTypePointer GetFragmentSetAddress(s32 delta_index, s32 fragment_set_index) const { auto header = this->GetHeader(); AMS_ABORT_UNLESS(static_cast(delta_index) < header->delta_count); auto delta_header = this->GetPatchDeltaHeader(delta_index); AMS_ABORT_UNLESS(static_cast(fragment_set_index) < delta_header->delta.fragment_set_count); auto previous_fragment_set_count = this->CountFragmentSet(delta_index); return this->GetHeaderAddress() + sizeof(PatchMetaExtendedDataHeader) + sizeof(PatchHistoryHeader) * header->history_count + sizeof(PatchDeltaHistory) * header->delta_history_count + sizeof(PatchDeltaHeader) * header->delta_count + sizeof(FragmentSet) * (previous_fragment_set_count + fragment_set_index); } DataTypePointer GetPatchHistoryContentInfoAddress(s32 history_index, s32 content_index) const { auto header = this->GetHeader(); auto history_header = this->GetPatchHistoryHeader(history_index); AMS_ABORT_UNLESS(static_cast(content_index) < history_header->content_count); auto prev_history_count = this->CountHistoryContent(history_index); return this->GetHeaderAddress() + sizeof(PatchMetaExtendedDataHeader) + sizeof(PatchHistoryHeader) * header->history_count + sizeof(PatchDeltaHistory) * header->delta_history_count + sizeof(PatchDeltaHeader) * header->delta_count + sizeof(FragmentSet) * header->fragment_set_count + sizeof(ContentInfo) * (prev_history_count + content_index); } DataTypePointer GetPatchDeltaPackagedContentInfoAddress(s32 delta_index, s32 content_index) const { auto header = this->GetHeader(); auto delta_header = this->GetPatchDeltaHeader(delta_index); AMS_ABORT_UNLESS(static_cast(content_index) < delta_header->content_count); auto content_count = this->CountDeltaContent(delta_index); return this->GetHeaderAddress() + sizeof(PatchMetaExtendedDataHeader) + sizeof(PatchHistoryHeader) * header->history_count + sizeof(PatchDeltaHistory) * header->delta_history_count + sizeof(PatchDeltaHeader) * header->delta_count + sizeof(FragmentSet) * header->fragment_set_count + sizeof(ContentInfo) * header->history_content_total_count + sizeof(PackagedContentInfo) * (content_count + content_index); } DataTypePointer GetFragmentIndicatorAddress(s32 delta_index, s32 fragment_set_index, s32 index) const { auto header = this->GetHeader(); auto fragment_set = this->GetFragmentSet(delta_index, fragment_set_index); AMS_ABORT_UNLESS(static_cast(index) < fragment_set->fragment_count); auto fragment_set_count = this->CountFragmentSet(delta_index); auto fragment_count = this->CountFragment(fragment_set_count + fragment_set_index); return this->GetHeaderAddress() + sizeof(PatchMetaExtendedDataHeader) + sizeof(PatchHistoryHeader) * header->history_count + sizeof(PatchDeltaHistory) * header->delta_history_count + sizeof(PatchDeltaHeader) * header->delta_count + sizeof(FragmentSet) * header->fragment_set_count + sizeof(ContentInfo) * header->history_content_total_count + sizeof(PackagedContentInfo) * header->delta_content_total_count + sizeof(FragmentIndicator) * (fragment_count + index); } public: const PatchMetaExtendedDataHeader *GetHeader() const { return reinterpret_cast(this->GetHeaderAddress()); } const PatchHistoryHeader *GetPatchHistoryHeader(s32 index) const { return reinterpret_cast(this->GetPatchHistoryHeaderAddress(index)); } const PatchDeltaHistory *GetPatchDeltaHistory(s32 index) const { return reinterpret_cast(this->GetPatchDeltaHistoryAddress(index)); } const ContentInfo *GetPatchHistoryContentInfo(s32 history_index, s32 content_index) const { return reinterpret_cast(this->GetPatchHistoryContentInfoAddress(history_index, content_index)); } const PatchDeltaHeader *GetPatchDeltaHeader(s32 index) const { return reinterpret_cast(this->GetPatchDeltaHeaderAddress(index)); } const PackagedContentInfo *GetPatchDeltaPackagedContentInfo(s32 delta_index, s32 content_index) const { return reinterpret_cast(this->GetPatchDeltaPackagedContentInfoAddress(delta_index, content_index)); } const FragmentSet *GetFragmentSet(s32 delta_index, s32 fragment_set_index) const { return reinterpret_cast(this->GetFragmentSetAddress(delta_index, fragment_set_index)); } const FragmentIndicator *GetFragmentIndicator(s32 delta_index, s32 fragment_set_index, s32 index) const { return reinterpret_cast(this->GetFragmentIndicatorAddress(delta_index, fragment_set_index, index)); } const FragmentIndicator *FindFragmentIndicator(s32 delta_index, s32 fragment_set_index, s32 fragment_index) const { auto fragment_set = this->GetFragmentSet(delta_index, fragment_set_index); auto fragment = this->GetFragmentIndicator(delta_index, fragment_set_index, 0); for (s32 i = 0; i < fragment_set->fragment_count; i++) { if (fragment[i].fragment_index == fragment_index) { return std::addressof(fragment[i]); } } return nullptr; } }; class PatchMetaExtendedDataReader : public PatchMetaExtendedDataReaderWriterBase { public: PatchMetaExtendedDataReader(const void *data, size_t size) : PatchMetaExtendedDataReaderWriterBase(data, size) { /* ... */ } }; class SystemUpdateMetaExtendedDataReaderWriterBase { private: void *m_data; const size_t m_size; bool m_is_header_valid; protected: constexpr SystemUpdateMetaExtendedDataReaderWriterBase(const void *d, size_t sz) : m_data(const_cast(d)), m_size(sz), m_is_header_valid(true) { /* ... */ } constexpr SystemUpdateMetaExtendedDataReaderWriterBase(void *d, size_t sz) : m_data(d), m_size(sz), m_is_header_valid(false) { /* ... */ } uintptr_t GetHeaderAddress() const { return reinterpret_cast(m_data); } uintptr_t GetFirmwareVariationIdStartAddress() const { return this->GetHeaderAddress() + sizeof(SystemUpdateMetaExtendedDataHeader); } uintptr_t GetFirmwareVariationIdAddress(size_t i) const { return this->GetFirmwareVariationIdStartAddress() + i * sizeof(FirmwareVariationId); } uintptr_t GetFirmwareVariationInfoStartAddress() const { return this->GetFirmwareVariationIdAddress(this->GetFirmwareVariationCount()); } uintptr_t GetFirmwareVariationInfoAddress(size_t i) const { return this->GetFirmwareVariationInfoStartAddress() + i * sizeof(FirmwareVariationInfo); } uintptr_t GetContentMetaInfoStartAddress() const { return this->GetFirmwareVariationInfoAddress(this->GetFirmwareVariationCount()); } uintptr_t GetContentMetaInfoAddress(size_t i) const { return this->GetContentMetaInfoStartAddress() + i * sizeof(ContentMetaInfo); } public: const SystemUpdateMetaExtendedDataHeader *GetHeader() const { AMS_ABORT_UNLESS(m_is_header_valid); return reinterpret_cast(this->GetHeaderAddress()); } size_t GetFirmwareVariationCount() const { return this->GetHeader()->firmware_variation_count; } const FirmwareVariationId *GetFirmwareVariationId(size_t i) const { AMS_ABORT_UNLESS(i < this->GetFirmwareVariationCount()); return reinterpret_cast(this->GetFirmwareVariationIdAddress(i)); } const FirmwareVariationInfo *GetFirmwareVariationInfo(size_t i) const { AMS_ABORT_UNLESS(i < this->GetFirmwareVariationCount()); return reinterpret_cast(this->GetFirmwareVariationInfoAddress(i)); } void GetContentMetaInfoList(Span *out_list, size_t i) const { size_t preceding_content_meta_count = 0; /* Count the number of preceding content metas. */ for (size_t j = 0; j < i; j++) { preceding_content_meta_count += this->GetFirmwareVariationInfo(j)->content_meta_count; } /* Output the list. */ *out_list = Span(reinterpret_cast(this->GetContentMetaInfoAddress(preceding_content_meta_count)), this->GetFirmwareVariationInfo(i)->content_meta_count); } }; class SystemUpdateMetaExtendedDataReader : public SystemUpdateMetaExtendedDataReaderWriterBase { public: constexpr SystemUpdateMetaExtendedDataReader(const void *data, size_t size) : SystemUpdateMetaExtendedDataReaderWriterBase(data, size) { /* ... */ } }; template class ReadableStructPin; class AccessorBase { public: template class PinBase { private: AccessorBase *m_accessor; u64 m_pin_id; T *m_data; size_t m_size; public: PinBase() : m_accessor(nullptr), m_data(nullptr), m_size(0) { /* ... */ } PinBase(const PinBase &) = delete; PinBase &operator=(const PinBase &) = delete; PinBase(PinBase &&rhs) : m_accessor(rhs.m_accessor), m_pin_id(rhs.m_pin_id), m_data(rhs.m_data), m_size(rhs.m_size) { rhs.m_accessor = nullptr; } PinBase &operator=(PinBase &&rhs) { m_accessor = rhs.m_accessor; m_pin_id = rhs.m_pin_id; m_data = rhs.m_data; m_size = rhs.m_size; rhs.m_accessor = nullptr; return *this; } virtual ~PinBase() { this->Reset(); } public: void Reset() { if (m_accessor != nullptr) { m_accessor->ReleasePin(m_pin_id); m_accessor = nullptr; } } void Reset(AccessorBase *accessor, u64 pin_id, void *data, size_t size) { AMS_ASSERT(data != nullptr || size == 0); this->Reset(); m_accessor = accessor; m_pin_id = pin_id; m_data = reinterpret_cast(data); m_size = size; } T *GetData() const { return m_data; } size_t GetDataSize() const { return m_size; } }; private: IMapper *m_mapper; public: AccessorBase(IMapper *mapper) : m_mapper(mapper) { /* ... */ } template Result AcquireReadableStructPin(ReadableStructPin *out, size_t offset) { /* Acquire mapped memory for the pin. */ MappedMemory memory = {}; R_TRY(m_mapper->GetMappedMemory(std::addressof(memory), offset, sizeof(T))); /* Mark the memory as in use. */ R_RETURN(m_mapper->MarkUsing(memory.id)); /* Setup the pin. */ out->Reset(this, memory.id, memory.GetBuffer(offset, sizeof(T)), sizeof(T)); R_SUCCEED(); } Result ReleasePin(u64 id) { R_RETURN(m_mapper->UnmarkUsing(id)); } template Result ReadStruct(T *out, size_t offset) { /* Acquire mapped memory for the pin. */ MappedMemory memory = {}; R_TRY(m_mapper->GetMappedMemory(std::addressof(memory), offset, sizeof(T))); /* Mark the memory as in use. */ R_RETURN(m_mapper->MarkUsing(memory.id)); ON_SCOPE_EXIT { this->ReleasePin(memory.id); }; /* Copy out the struct. */ *out = *reinterpret_cast(memory.GetBuffer(offset, sizeof(T))); R_SUCCEED(); } }; template class ReadableStructPin final : public AccessorBase::PinBase { public: using PinBase::PinBase; using PinBase::operator=; const T *Get() const { return reinterpret_cast(this->GetData()); } size_t GetSize() const { return this->GetDataSize(); } const T &operator*() const { return *this->Get(); } const T *operator->() const { return this->Get(); } }; class PatchMetaExtendedDataAccessor : public AccessorBase { private: struct CachedCount { s32 index; s32 count; }; private: util::optional m_cached_history_content_count = util::nullopt; util::optional m_cached_delta_content_count = util::nullopt; util::optional m_cached_fragment_set_count = util::nullopt; util::optional m_cached_fragment_indicator_count = util::nullopt; util::optional m_header = util::nullopt; public: using AccessorBase::AccessorBase; public: Result GetHeader(ReadableStructPin *out) { R_RETURN(this->AcquireReadableStructPin(out, 0)); } Result GetHeader(PatchMetaExtendedDataHeader *out) { R_RETURN(this->template ReadStruct(out, 0)); } Result GetHistoryHeader(ReadableStructPin *out, s32 index) { /* Ensure we have our header. */ R_TRY(this->EnsureHeader()); /* Check that the index is valid. */ R_UNLESS(0 <= index && static_cast(index) < m_header->history_count, ncm::ResultInvalidOffset()); /* Get the header. */ const size_t offset = sizeof(PatchHistoryHeader) + sizeof(PatchHistoryHeader) * index; R_RETURN(this->AcquireReadableStructPin(out, offset)); } Result GetPatchDeltaHistory(ReadableStructPin *out, s32 index) { /* Ensure we have our header. */ R_TRY(this->EnsureHeader()); /* Check that the index is valid. */ R_UNLESS(0 <= index && static_cast(index) < m_header->delta_history_count, ncm::ResultInvalidOffset()); /* Get the history. */ const size_t offset = sizeof(PatchHistoryHeader) + sizeof(PatchHistoryHeader) * m_header->history_count + sizeof(PatchDeltaHistory) * index; R_RETURN(this->AcquireReadableStructPin(out, offset)); } Result GetPatchDeltaHeader(ReadableStructPin *out, s32 index) { /* Ensure we have our header. */ R_TRY(this->EnsureHeader()); /* Check that the index is valid. */ R_UNLESS(0 <= index && static_cast(index) < m_header->delta_count, ncm::ResultInvalidOffset()); /* Get the header. */ const size_t offset = sizeof(PatchHistoryHeader) + sizeof(PatchHistoryHeader) * m_header->history_count + sizeof(PatchDeltaHistory) * m_header->delta_history_count + sizeof(PatchDeltaHeader) * index; R_RETURN(this->AcquireReadableStructPin(out, offset)); } Result GetFragmentSet(ReadableStructPin *out, s32 delta_index, s32 fragment_set_index) { /* Ensure we have our header. */ R_TRY(this->EnsureHeader()); /* Check that the index is valid. */ R_UNLESS(0 <= delta_index && static_cast(delta_index) < m_header->delta_count, ncm::ResultInvalidOffset()); /* Get the previous fragment set count. */ s32 previous_fragment_set_count = 0; R_TRY(this->CountFragmentSet(std::addressof(previous_fragment_set_count), delta_index)); /* Get the set. */ const size_t offset = sizeof(PatchHistoryHeader) + sizeof(PatchHistoryHeader) * m_header->history_count + sizeof(PatchDeltaHistory) * m_header->delta_history_count + sizeof(PatchDeltaHeader) * m_header->delta_count + sizeof(FragmentSet) * (previous_fragment_set_count + fragment_set_index); R_RETURN(this->AcquireReadableStructPin(out, offset)); } Result GetFragmentSetDirectly(ReadableStructPin *out, s32 fragment_set_direct_index) { /* Ensure we have our header. */ R_TRY(this->EnsureHeader()); /* Check that the index is valid. */ R_UNLESS(0 <= fragment_set_direct_index && static_cast(fragment_set_direct_index) < m_header->fragment_set_count, ncm::ResultInvalidOffset()); /* Get the set. */ const size_t offset = sizeof(PatchHistoryHeader) + sizeof(PatchHistoryHeader) * m_header->history_count + sizeof(PatchDeltaHistory) * m_header->delta_history_count + sizeof(PatchDeltaHeader) * m_header->delta_count + sizeof(FragmentSet) * (fragment_set_direct_index); R_RETURN(this->AcquireReadableStructPin(out, offset)); } Result GetPatchHistoryContentInfo(ReadableStructPin *out, s32 history_index, s32 content_index) { /* Ensure we have our header. */ R_TRY(this->EnsureHeader()); /* Check that the index is valid. */ R_UNLESS(0 <= history_index && static_cast(history_index) < m_header->history_count, ncm::ResultInvalidOffset()); /* Determine the true history content index. */ s32 prev_history_count = 0; R_TRY(this->CountHistoryContentInfo(std::addressof(prev_history_count), history_index)); /* Adjust and check the content index. */ content_index += prev_history_count; R_UNLESS(0 <= content_index && static_cast(content_index) < m_header->history_content_total_count, ncm::ResultInvalidOffset()); /* Get the info. */ const size_t offset = sizeof(PatchHistoryHeader) + sizeof(PatchHistoryHeader) * m_header->history_count + sizeof(PatchDeltaHistory) * m_header->delta_history_count + sizeof(PatchDeltaHeader) * m_header->delta_count + sizeof(FragmentSet) * m_header->fragment_set_count + sizeof(ContentInfo) * content_index; R_RETURN(this->AcquireReadableStructPin(out, offset)); } Result GetPatchDeltaContentInfo(ReadableStructPin *out, s32 delta_index, s32 content_index) { /* Ensure we have our header. */ R_TRY(this->EnsureHeader()); /* Check that the index is valid. */ R_UNLESS(0 <= delta_index && static_cast(delta_index) < m_header->delta_count, ncm::ResultInvalidOffset()); /* Determine the true delta content index. */ s32 prev_delta_count = 0; R_TRY(this->CountDeltaContentInfo(std::addressof(prev_delta_count), delta_index)); /* Adjust and check the content index. */ content_index += prev_delta_count; R_UNLESS(0 <= content_index && static_cast(content_index) < m_header->delta_content_total_count, ncm::ResultInvalidOffset()); /* Get the info. */ const size_t offset = sizeof(PatchHistoryHeader) + sizeof(PatchHistoryHeader) * m_header->history_count + sizeof(PatchDeltaHistory) * m_header->delta_history_count + sizeof(PatchDeltaHeader) * m_header->delta_count + sizeof(FragmentSet) * m_header->fragment_set_count + sizeof(ContentInfo) * m_header->history_content_total_count + sizeof(PackagedContentInfo) * content_index; R_RETURN(this->AcquireReadableStructPin(out, offset)); } Result GetFragmentIndicator(ReadableStructPin *out, s32 delta_index, s32 fragment_set_index, s32 index) { /* Ensure we have our header. */ R_TRY(this->EnsureHeader()); /* Check that the index is valid. */ R_UNLESS(0 <= delta_index && static_cast(delta_index) < m_header->delta_count, ncm::ResultInvalidOffset()); /* Get the previous fragment set count. */ s32 previous_fragment_set_count = 0; R_TRY(this->CountFragmentSet(std::addressof(previous_fragment_set_count), delta_index)); /* Get the previous fragment indicator count. */ s32 previous_fragment_count = 0; R_TRY(this->CountFragmentIndicator(std::addressof(previous_fragment_count), previous_fragment_count + fragment_set_index)); /* Get the info. */ const size_t offset = sizeof(PatchHistoryHeader) + sizeof(PatchHistoryHeader) * m_header->history_count + sizeof(PatchDeltaHistory) * m_header->delta_history_count + sizeof(PatchDeltaHeader) * m_header->delta_count + sizeof(FragmentSet) * m_header->fragment_set_count + sizeof(ContentInfo) * m_header->history_content_total_count + sizeof(PackagedContentInfo) * m_header->delta_content_total_count + sizeof(FragmentIndicator) * (previous_fragment_count + index); R_RETURN(this->AcquireReadableStructPin(out, offset)); } Result FindFragmentIndicator(ReadableStructPin *out, s32 delta_index, s32 fragment_set_index, s32 fragment_index) { /* Ensure we have our header. */ R_TRY(this->EnsureHeader()); /* Check that the index is valid. */ R_UNLESS(0 <= delta_index && static_cast(delta_index) < m_header->delta_count, ncm::ResultInvalidOffset()); /* Get the fragment count. */ s32 fragment_count = 0; { ReadableStructPin set; R_TRY(this->GetFragmentSet(std::addressof(set), delta_index, fragment_set_index)); fragment_count = set->fragment_count; } /* Get the previous fragment set count. */ s32 previous_fragment_set_count = 0; R_TRY(this->CountFragmentSet(std::addressof(previous_fragment_set_count), delta_index)); /* Get the previous fragment indicator count. */ s32 previous_fragment_count = 0; R_TRY(this->CountFragmentIndicator(std::addressof(previous_fragment_count), previous_fragment_count + fragment_set_index)); /* Look for a correct indicator. */ for (auto i = 0; i < fragment_count; ++i) { /* Get the current info. */ ReadableStructPin indicator; const size_t offset = sizeof(PatchHistoryHeader) + sizeof(PatchHistoryHeader) * m_header->history_count + sizeof(PatchDeltaHistory) * m_header->delta_history_count + sizeof(PatchDeltaHeader) * m_header->delta_count + sizeof(FragmentSet) * m_header->fragment_set_count + sizeof(ContentInfo) * m_header->history_content_total_count + sizeof(PackagedContentInfo) * m_header->delta_content_total_count + sizeof(FragmentIndicator) * (previous_fragment_count + i); R_TRY(this->AcquireReadableStructPin(std::addressof(indicator), offset)); /* If it matches, return it. */ if (indicator->fragment_index == fragment_index) { *out = std::move(indicator); R_SUCCEED(); } } /* We didn't find an indicator. */ R_THROW(ncm::ResultFragmentIndicatorNotFound()); } Result GetHistoryHeader(PatchHistoryHeader *out, s32 index) { /* Get the pin. */ ReadableStructPin pin; R_TRY(this->GetHistoryHeader(std::addressof(pin), index)); /* Copy it out. */ *out = *pin; R_SUCCEED(); } Result GetPatchDeltaHistory(PatchDeltaHistory *out, s32 index) { /* Get the pin. */ ReadableStructPin pin; R_TRY(this->GetPatchDeltaHistory(std::addressof(pin), index)); /* Copy it out. */ *out = *pin; R_SUCCEED(); } Result GetPatchDeltaHeader(PatchDeltaHeader *out, s32 index) { /* Get the pin. */ ReadableStructPin pin; R_TRY(this->GetPatchDeltaHeader(std::addressof(pin), index)); /* Copy it out. */ *out = *pin; R_SUCCEED(); } Result GetFragmentSet(FragmentSet *out, s32 delta_index, s32 fragment_set_index) { /* Get the pin. */ ReadableStructPin pin; R_TRY(this->GetFragmentSet(std::addressof(pin), delta_index, fragment_set_index)); /* Copy it out. */ *out = *pin; R_SUCCEED(); } Result GetPatchHistoryContentInfo(ContentInfo *out, s32 history_index, s32 content_index) { /* Get the header. */ ReadableStructPin pin; R_TRY(this->GetPatchHistoryContentInfo(std::addressof(pin), history_index, content_index)); /* Copy it out. */ *out = *pin; R_SUCCEED(); } Result GetPatchDeltaContentInfo(PackagedContentInfo *out, s32 delta_index, s32 content_index) { /* Get the header. */ ReadableStructPin pin; R_TRY(this->GetPatchDeltaContentInfo(std::addressof(pin), delta_index, content_index)); /* Copy it out. */ *out = *pin; R_SUCCEED(); } Result GetFragmentIndicator(FragmentIndicator *out, s32 delta_index, s32 fragment_set_index, s32 index) { /* Get the header. */ ReadableStructPin pin; R_TRY(this->GetFragmentIndicator(std::addressof(pin), delta_index, fragment_set_index, index)); /* Copy it out. */ *out = *pin; R_SUCCEED(); } Result FindFragmentIndicator(FragmentIndicator *out, s32 delta_index, s32 fragment_set_index, s32 fragment_index) { /* Get the header. */ ReadableStructPin pin; R_TRY(this->FindFragmentIndicator(std::addressof(pin), delta_index, fragment_set_index, fragment_index)); /* Copy it out. */ *out = *pin; R_SUCCEED(); } Result CountHistoryContentInfo(s32 *out, s32 index) { R_RETURN(this->CountImpl(out, index, m_cached_history_content_count, [&](s32 *out, s32 i) -> Result { /* Get the history header. */ ReadableStructPin header; R_TRY(this->GetHistoryHeader(std::addressof(header), i)); /* Set the content count. */ *out = header->content_count; R_SUCCEED(); })); } Result CountDeltaContentInfo(s32 *out, s32 index) { R_RETURN(this->CountImpl(out, index, m_cached_delta_content_count, [&](s32 *out, s32 i) -> Result { /* Get the history header. */ ReadableStructPin header; R_TRY(this->GetPatchDeltaHeader(std::addressof(header), i)); /* Set the content count. */ *out = header->content_count; R_SUCCEED(); })); } Result CountFragmentSet(s32 *out, s32 index) { R_RETURN(this->CountImpl(out, index, m_cached_fragment_set_count, [&](s32 *out, s32 i) -> Result { /* Get the history header. */ ReadableStructPin header; R_TRY(this->GetPatchDeltaHeader(std::addressof(header), i)); /* Set the fragment set count. */ *out = header->delta.fragment_set_count; R_SUCCEED(); })); } Result CountFragmentIndicator(s32 *out, s32 index) { R_RETURN(this->CountImpl(out, index, m_cached_fragment_indicator_count, [&](s32 *out, s32 i) -> Result { /* Get the history header. */ ReadableStructPin set; R_TRY(this->GetFragmentSetDirectly(std::addressof(set), i)); /* Set the indicator count. */ *out = set->fragment_count; R_SUCCEED(); })); } private: Result CountImpl(s32 *out, s32 index, util::optional &cache, auto get_count_impl) const { /* Ensure the value is cached. */ if (!(cache.has_value() && cache->index == index)) { /* Determine the count. */ CachedCount calc = { .index = index, .count = 0 }; for (auto i = 0; i < index; ++i) { s32 cur_count = 0; R_TRY(get_count_impl(std::addressof(cur_count), i)); calc.count += cur_count; } /* Cache the count. */ cache = calc; } /* Set the output count. */ *out = cache->count; R_SUCCEED(); } private: Result EnsureHeader() { /* If we have our header, we're good. */ R_SUCCEED_IF(m_header.has_value()); /* Get our header. */ PatchMetaExtendedDataHeader header{}; R_TRY(this->GetHeader(std::addressof(header))); /* Set our header. */ m_header.emplace(header); R_SUCCEED(); } }; }