/* * 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/>. */ #pragma once #include <stratosphere.hpp> #include "sysupdater_thread_allocator.hpp" namespace ams::mitm::sysupdater { class ErrorContextHolder { private: err::ErrorContext m_error_context; public: constexpr ErrorContextHolder() : m_error_context{} { /* ... */ } virtual ~ErrorContextHolder() { /* ... */ } template<typename T> Result SaveErrorContextIfFailed(T &async, Result result) { ON_RESULT_FAILURE { async.GetErrorContext(std::addressof(m_error_context)); }; R_RETURN(result); } template<typename T> Result GetAndSaveErrorContext(T &async) { R_RETURN(this->SaveErrorContextIfFailed(async, async.Get())); } template<typename T> Result SaveInternalTaskErrorContextIfFailed(T &async, Result result) { ON_RESULT_FAILURE { async.CreateErrorContext(std::addressof(m_error_context)); }; R_RETURN(result); } const err::ErrorContext &GetErrorContextImpl() { return m_error_context; } }; class AsyncBase { public: virtual ~AsyncBase() { /* ... */ } static Result ToAsyncResult(Result result); Result Cancel() { this->CancelImpl(); R_SUCCEED(); } virtual Result GetErrorContext(sf::Out<err::ErrorContext> out) { *out = {}; R_SUCCEED(); } private: virtual void CancelImpl() = 0; }; class AsyncResultBase : public AsyncBase { public: virtual ~AsyncResultBase() { /* ... */ } Result Get() { R_RETURN(ToAsyncResult(this->GetImpl())); } private: virtual Result GetImpl() = 0; }; static_assert(ns::impl::IsIAsyncResult<AsyncResultBase>); /* NOTE: Based off of ns AsyncPrepareCardUpdateImpl. */ /* We don't implement the RequestServer::ManagedStop details, as we don't implement stoppable request list. */ class AsyncPrepareSdCardUpdateImpl : public AsyncResultBase, private ErrorContextHolder { private: Result m_result; os::SystemEvent m_event; util::optional<ThreadInfo> m_thread_info; ncm::InstallTaskBase *m_task; public: AsyncPrepareSdCardUpdateImpl(ncm::InstallTaskBase *task) : m_result(ResultSuccess()), m_event(os::EventClearMode_ManualClear, true), m_thread_info(), m_task(task) { /* ... */ } virtual ~AsyncPrepareSdCardUpdateImpl(); os::SystemEvent &GetEvent() { return m_event; } virtual Result GetErrorContext(sf::Out<err::ErrorContext> out) override { *out = ErrorContextHolder::GetErrorContextImpl(); R_SUCCEED(); } Result Run(); private: Result Execute(); virtual void CancelImpl() override; virtual Result GetImpl() override { return m_result; } }; }