/* * 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 <http://www.gnu.org/licenses/>. */ #include <stratosphere.hpp> #include "sysupdater_apply_manager.hpp" namespace ams::mitm::sysupdater { namespace { alignas(os::MemoryPageSize) u8 g_boot_image_update_buffer[64_KB]; updater::BootImageUpdateType GetBootImageUpdateType() { /* NOTE: Here Nintendo uses the value of system setting systeminitializer!boot_image_update_type...but we prefer not to take the risk. */ return updater::GetBootImageUpdateType(spl::GetHardwareType()); } Result MarkPreCommitForBootImages() { /* Set verification required for both normal and safe mode. */ R_TRY(updater::MarkVerifyingRequired(updater::BootModeType::Normal, g_boot_image_update_buffer, sizeof(g_boot_image_update_buffer))); R_TRY(updater::MarkVerifyingRequired(updater::BootModeType::Safe, g_boot_image_update_buffer, sizeof(g_boot_image_update_buffer))); /* Pre-commit is now marked. */ R_SUCCEED(); } Result UpdateBootImages() { /* Define a helper to update the images. */ auto UpdateBootImageImpl = [](updater::BootModeType boot_mode, updater::BootImageUpdateType boot_image_update_type) -> Result { /* Get the boot image package id. */ ncm::SystemDataId boot_image_package_id = {}; R_TRY_CATCH(updater::GetBootImagePackageId(std::addressof(boot_image_package_id), boot_mode, g_boot_image_update_buffer, sizeof(g_boot_image_update_buffer))) { R_CATCH(updater::ResultBootImagePackageNotFound) { /* Nintendo simply falls through when the package is not found. */ } } R_END_TRY_CATCH; /* Update the boot images. */ R_TRY_CATCH(updater::UpdateBootImagesFromPackage(boot_image_package_id, boot_mode, g_boot_image_update_buffer, sizeof(g_boot_image_update_buffer), boot_image_update_type)) { R_CATCH(updater::ResultBootImagePackageNotFound) { /* Nintendo simply falls through when the package is not found. */ } } R_END_TRY_CATCH; /* Mark the images verified. */ R_TRY(updater::MarkVerified(boot_mode, g_boot_image_update_buffer, sizeof(g_boot_image_update_buffer))); /* The boot images are updated. */ R_SUCCEED(); }; /* Get the boot image update type. */ auto boot_image_update_type = GetBootImageUpdateType(); /* Update boot images for safe mode. */ R_TRY(UpdateBootImageImpl(updater::BootModeType::Safe, boot_image_update_type)); /* Update boot images for normal mode. */ R_TRY(UpdateBootImageImpl(updater::BootModeType::Normal, boot_image_update_type)); /* Both sets of images are updated. */ R_SUCCEED(); } } Result SystemUpdateApplyManager::ApplyPackageTask(ncm::PackageSystemDowngradeTask *task) { /* Lock the apply mutex. */ std::scoped_lock lk(m_apply_mutex); /* NOTE: Here, Nintendo creates a system report for the update. */ /* Mark boot images to note that we're updating. */ R_TRY(MarkPreCommitForBootImages()); /* Commit the task. */ R_TRY(task->Commit()); /* Update the boot images. */ R_TRY(UpdateBootImages()); R_SUCCEED(); } }