2020-06-26 04:59:59 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018-2020 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>
|
|
|
|
#include "sysupdater_service.hpp"
|
2020-06-26 12:05:24 +00:00
|
|
|
#include "sysupdater_fs_utils.hpp"
|
2020-06-26 04:59:59 +00:00
|
|
|
|
|
|
|
namespace ams::mitm::sysupdater {
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
/* ExFat NCAs prior to 2.0.0 do not actually include the exfat driver, and don't boot. */
|
|
|
|
constexpr inline u32 MinimumVersionForExFatDriver = 65536;
|
|
|
|
|
2020-06-26 12:05:24 +00:00
|
|
|
template<typename F>
|
|
|
|
Result ForEachFileInDirectory(const char *root_path, F f) {
|
|
|
|
/* Open the directory. */
|
|
|
|
fs::DirectoryHandle dir;
|
|
|
|
R_TRY(fs::OpenDirectory(std::addressof(dir), root_path, fs::OpenDirectoryMode_File));
|
|
|
|
ON_SCOPE_EXIT { fs::CloseDirectory(dir); };
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
/* Read the current entry. */
|
|
|
|
s64 count;
|
|
|
|
fs::DirectoryEntry entry;
|
|
|
|
R_TRY(fs::ReadDirectory(std::addressof(count), std::addressof(entry), dir, 1));
|
|
|
|
if (count == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Invoke our handler on the entry. */
|
|
|
|
bool done;
|
|
|
|
R_TRY(f(std::addressof(done), entry));
|
|
|
|
R_SUCCEED_IF(done);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
Result ConvertToFsCommonPath(char *dst, size_t dst_size, const char *package_root_path, const char *entry_path) {
|
|
|
|
char package_path[ams::fs::EntryNameLengthMax];
|
|
|
|
|
|
|
|
const size_t path_len = std::snprintf(package_path, sizeof(package_path), "%s%s", package_root_path, entry_path);
|
|
|
|
AMS_ABORT_UNLESS(path_len < ams::fs::EntryNameLengthMax);
|
|
|
|
|
|
|
|
return ams::fs::ConvertToFsCommonPath(dst, dst_size, package_path);
|
|
|
|
}
|
|
|
|
|
|
|
|
Result LoadContentMeta(ncm::AutoBuffer *out, const char *package_root_path, const fs::DirectoryEntry &entry) {
|
|
|
|
AMS_ABORT_UNLESS(PathView(entry.name).HasSuffix(".cnmt.nca"));
|
|
|
|
|
|
|
|
char path[ams::fs::EntryNameLengthMax];
|
|
|
|
R_TRY(ConvertToFsCommonPath(path, sizeof(path), package_root_path, entry.name));
|
|
|
|
|
|
|
|
return ncm::ReadContentMetaPath(out, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
Result ReadContentMetaPath(ncm::AutoBuffer *out, const char *package_root, const ncm::ContentInfo &content_info) {
|
|
|
|
/* Get the .cnmt.nca path for the info. */
|
|
|
|
char cnmt_nca_name[ncm::ContentIdStringLength + 10];
|
|
|
|
ncm::GetStringFromContentId(cnmt_nca_name, sizeof(cnmt_nca_name), content_info.GetId());
|
|
|
|
std::memcpy(cnmt_nca_name + ncm::ContentIdStringLength, ".cnmt.nca", std::strlen(".cnmt.nca"));
|
|
|
|
cnmt_nca_name[sizeof(cnmt_nca_name) - 1] = '\x00';
|
|
|
|
|
|
|
|
/* Create a new path. */
|
|
|
|
ncm::Path content_path;
|
|
|
|
R_TRY(ConvertToFsCommonPath(content_path.str, sizeof(content_path.str), package_root, cnmt_nca_name));
|
|
|
|
|
|
|
|
/* Read the content meta path. */
|
|
|
|
return ncm::ReadContentMetaPath(out, content_path.str);
|
|
|
|
}
|
|
|
|
|
|
|
|
Result GetSystemUpdateUpdateContentInfoFromPackage(ncm::ContentInfo *out, const char *package_root) {
|
|
|
|
bool found_system_update = false;
|
|
|
|
|
|
|
|
/* Iterate over all files to find the system update meta. */
|
|
|
|
R_TRY(ForEachFileInDirectory(package_root, [&](bool *done, const fs::DirectoryEntry &entry) -> Result {
|
|
|
|
/* Don't early terminate by default. */
|
|
|
|
*done = false;
|
|
|
|
|
|
|
|
/* We have nothing to list if we're not looking at a meta. */
|
|
|
|
R_SUCCEED_IF(!PathView(entry.name).HasSuffix(".cnmt.nca"));
|
|
|
|
|
|
|
|
/* Read the content meta path, and build. */
|
|
|
|
ncm::AutoBuffer package_meta;
|
|
|
|
R_TRY(LoadContentMeta(std::addressof(package_meta), package_root, entry));
|
|
|
|
|
|
|
|
/* Create a reader. */
|
|
|
|
const auto reader = ncm::PackagedContentMetaReader(package_meta.Get(), package_meta.GetSize());
|
|
|
|
|
|
|
|
/* If we find a system update, we're potentially done. */
|
|
|
|
if (reader.GetHeader()->type == ncm::ContentMetaType::SystemUpdate) {
|
|
|
|
/* Try to parse a content id from the name. */
|
|
|
|
auto content_id = ncm::GetContentIdFromString(entry.name, sizeof(entry.name));
|
|
|
|
R_UNLESS(content_id, ncm::ResultInvalidPackageFormat());
|
|
|
|
|
|
|
|
/* We're done. */
|
|
|
|
*done = true;
|
|
|
|
found_system_update = true;
|
|
|
|
|
|
|
|
*out = ncm::ContentInfo::Make(*content_id, entry.file_size, ncm::ContentType::Meta);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ResultSuccess();
|
|
|
|
}));
|
|
|
|
|
|
|
|
/* If we didn't find anything, error. */
|
|
|
|
R_UNLESS(found_system_update, ncm::ResultSystemUpdateNotFoundInPackage());
|
|
|
|
|
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
2020-06-26 04:59:59 +00:00
|
|
|
Result ActivateSystemUpdateContentMetaDatabase() {
|
|
|
|
/* TODO: Don't use gamecard db. */
|
|
|
|
return ncm::ActivateContentMetaDatabase(ncm::StorageId::GameCard);
|
|
|
|
}
|
|
|
|
|
|
|
|
void InactivateSystemUpdateContentMetaDatabase() {
|
|
|
|
/* TODO: Don't use gamecard db. */
|
|
|
|
ncm::InactivateContentMetaDatabase(ncm::StorageId::GameCard);
|
|
|
|
}
|
|
|
|
|
|
|
|
Result OpenSystemUpdateContentMetaDatabase(ncm::ContentMetaDatabase *out) {
|
|
|
|
/* TODO: Don't use gamecard db. */
|
|
|
|
return ncm::OpenContentMetaDatabase(out, ncm::StorageId::GameCard);
|
|
|
|
}
|
|
|
|
|
|
|
|
Result GetContentInfoOfContentMeta(ncm::ContentInfo *out, ncm::ContentMetaDatabase &db, const ncm::ContentMetaKey &key) {
|
|
|
|
s32 ofs = 0;
|
|
|
|
while (true) {
|
|
|
|
/* List content infos. */
|
|
|
|
s32 count;
|
|
|
|
ncm::ContentInfo info;
|
|
|
|
R_TRY(db.ListContentInfo(std::addressof(count), std::addressof(info), 1, key, ofs++));
|
|
|
|
|
|
|
|
/* No content infos left to list. */
|
|
|
|
if (count == 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if the info is for meta content. */
|
|
|
|
if (info.GetType() == ncm::ContentType::Meta) {
|
|
|
|
*out = info;
|
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Not found. */
|
|
|
|
return ncm::ResultContentInfoNotFound();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsExFatDriverSupported(const ncm::ContentMetaInfo &info) {
|
|
|
|
return info.version >= MinimumVersionForExFatDriver && ((info.attributes & ncm::ContentMetaAttribute_IncludesExFatDriver) != 0);
|
|
|
|
}
|
|
|
|
|
2020-06-26 11:34:26 +00:00
|
|
|
Result FormatUserPackagePath(ncm::Path *out, const ncm::Path &user_path) {
|
|
|
|
/* Ensure that the user path is valid. */
|
|
|
|
R_UNLESS(user_path.str[0] == '/', fs::ResultInvalidPath());
|
|
|
|
|
|
|
|
/* Print as @Sdcard:<user_path>/ */
|
|
|
|
std::snprintf(out->str, sizeof(out->str), "%s:%s/", ams::fs::impl::SdCardFileSystemMountName, user_path.str);
|
|
|
|
|
|
|
|
/* Normalize, if the user provided an ending / */
|
|
|
|
const size_t len = std::strlen(out->str);
|
|
|
|
if (out->str[len - 1] == '/' && out->str[len - 2] == '/') {
|
|
|
|
out->str[len - 1] = '\x00';
|
|
|
|
}
|
|
|
|
|
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
2020-06-26 04:59:59 +00:00
|
|
|
}
|
|
|
|
|
2020-06-26 11:34:26 +00:00
|
|
|
Result SystemUpdateService::GetUpdateInformation(sf::Out<UpdateInformation> out, const ncm::Path &path) {
|
|
|
|
/* Adjust the path. */
|
|
|
|
ncm::Path package_root;
|
|
|
|
R_TRY(FormatUserPackagePath(std::addressof(package_root), path));
|
|
|
|
|
2020-06-26 04:59:59 +00:00
|
|
|
/* Create a new update information. */
|
|
|
|
UpdateInformation update_info = {};
|
|
|
|
|
|
|
|
/* Parse the update. */
|
|
|
|
{
|
2020-06-26 12:05:24 +00:00
|
|
|
/* Get the content info for the system update. */
|
2020-06-26 04:59:59 +00:00
|
|
|
ncm::ContentInfo content_info;
|
2020-06-26 12:05:24 +00:00
|
|
|
R_TRY(GetSystemUpdateUpdateContentInfoFromPackage(std::addressof(content_info), package_root.str));
|
2020-06-26 04:59:59 +00:00
|
|
|
|
|
|
|
/* Read the content meta. */
|
|
|
|
ncm::AutoBuffer content_meta_buffer;
|
|
|
|
R_TRY(ReadContentMetaPath(std::addressof(content_meta_buffer), package_root.str, content_info));
|
|
|
|
|
|
|
|
/* Create a reader. */
|
|
|
|
const auto reader = ncm::PackagedContentMetaReader(content_meta_buffer.Get(), content_meta_buffer.GetSize());
|
|
|
|
|
2020-06-26 11:34:26 +00:00
|
|
|
/* Get the version from the header. */
|
|
|
|
update_info.version = reader.GetHeader()->version;
|
|
|
|
|
2020-06-26 04:59:59 +00:00
|
|
|
/* Iterate over infos to find the system update info. */
|
|
|
|
for (size_t i = 0; i < reader.GetContentMetaCount(); ++i) {
|
|
|
|
const auto &meta_info = *reader.GetContentMetaInfo(i);
|
|
|
|
|
|
|
|
switch (meta_info.type) {
|
|
|
|
case ncm::ContentMetaType::BootImagePackage:
|
|
|
|
/* Detect exFAT support. */
|
|
|
|
update_info.exfat_supported |= IsExFatDriverSupported(meta_info);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Default to no firmware variations. */
|
|
|
|
update_info.firmware_variation_count = 0;
|
|
|
|
|
|
|
|
/* Parse firmware variations if relevant. */
|
|
|
|
if (reader.GetExtendedDataSize() != 0) {
|
|
|
|
/* Get the actual firmware variation count. */
|
|
|
|
ncm::SystemUpdateMetaExtendedDataReader extended_data_reader(reader.GetExtendedData(), reader.GetExtendedDataSize());
|
|
|
|
update_info.firmware_variation_count = extended_data_reader.GetFirmwareVariationCount();
|
|
|
|
|
|
|
|
/* NOTE: Update this if Nintendo ever actually releases an update with this many variations? */
|
|
|
|
R_UNLESS(update_info.firmware_variation_count <= FirmwareVariationCountMax, ncm::ResultInvalidFirmwareVariation());
|
|
|
|
|
|
|
|
for (size_t i = 0; i < update_info.firmware_variation_count; ++i) {
|
|
|
|
update_info.firmware_variation_ids[i] = *extended_data_reader.GetFirmwareVariationId(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set the parsed update info. */
|
|
|
|
out.SetValue(update_info);
|
|
|
|
return ResultSuccess();
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|