/* * Copyright (c) 2019-2020 Adubbz, 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 <http://www.gnu.org/licenses/>. */ #include <stratosphere.hpp> namespace ams::ncm { namespace { constexpr inline s64 EncryptionMetadataSize = 16_KB; constexpr inline s64 ConcatenationFileSizeMax = 4_GB; constexpr s64 CalculateAdditionalContentSize(s64 file_size, s64 cluster_size) { /* Account for the encryption header. */ s64 size = EncryptionMetadataSize; /* Account for the file size splitting costs. */ size += ((file_size / ConcatenationFileSizeMax) + 1) * cluster_size; /* Account for various overhead costs. */ size += cluster_size * 3; return size; } template<typename Handler> Result ForEachContentInfo(const ContentMetaKey &key, ncm::ContentMetaDatabase *db, Handler handler) { constexpr s32 MaxPerIteration = 0x10; ContentInfo info_list[MaxPerIteration]; s32 offset = 0; while (true) { /* List the content infos. */ s32 count; R_TRY(db->ListContentInfo(std::addressof(count), info_list, MaxPerIteration, key, offset)); /* Handle all that we listed. */ for (s32 i = 0; i < count; i++) { bool done = false; R_TRY(handler(std::addressof(done), info_list[i])); if (done) { break; } } /* Check if we're done. */ if (count != MaxPerIteration) { break; } offset += count; } return ResultSuccess(); } } s64 CalculateRequiredSize(s64 file_size, s64 cluster_size) { return file_size + CalculateAdditionalContentSize(file_size, cluster_size); } s64 CalculateRequiredSizeForExtension(s64 file_size, s64 cluster_size) { return file_size + ((file_size / ConcatenationFileSizeMax) + 1) * cluster_size; } Result EstimateRequiredSize(s64 *out_size, const ContentMetaKey &key, ncm::ContentMetaDatabase *db) { s64 size = 0; R_TRY(ForEachContentInfo(key, db, [&size](bool *out_done, const ContentInfo &info) -> Result { size += CalculateRequiredSize(info.GetSize(), MaxClusterSize); *out_done = false; return ResultSuccess(); })); *out_size = size; return ResultSuccess(); } }