early-access version 4162
This commit is contained in:
		| @@ -1,7 +1,7 @@ | |||||||
| yuzu emulator early access | yuzu emulator early access | ||||||
| ============= | ============= | ||||||
|  |  | ||||||
| This is the source code for early-access 4161. | This is the source code for early-access 4162. | ||||||
|  |  | ||||||
| ## Legal Notice | ## Legal Notice | ||||||
|  |  | ||||||
|   | |||||||
| @@ -486,8 +486,10 @@ add_library(core STATIC | |||||||
|     hle/service/am/service/system_applet_proxy.h |     hle/service/am/service/system_applet_proxy.h | ||||||
|     hle/service/am/service/window_controller.cpp |     hle/service/am/service/window_controller.cpp | ||||||
|     hle/service/am/service/window_controller.h |     hle/service/am/service/window_controller.h | ||||||
|     hle/service/aoc/aoc_u.cpp |     hle/service/aoc/addon_content_manager.cpp | ||||||
|     hle/service/aoc/aoc_u.h |     hle/service/aoc/addon_content_manager.h | ||||||
|  |     hle/service/aoc/purchase_event_manager.cpp | ||||||
|  |     hle/service/aoc/purchase_event_manager.h | ||||||
|     hle/service/apm/apm.cpp |     hle/service/apm/apm.cpp | ||||||
|     hle/service/apm/apm.h |     hle/service/apm/apm.h | ||||||
|     hle/service/apm/apm_controller.cpp |     hle/service/apm/apm_controller.cpp | ||||||
|   | |||||||
							
								
								
									
										223
									
								
								src/core/hle/service/aoc/addon_content_manager.cpp
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										223
									
								
								src/core/hle/service/aoc/addon_content_manager.cpp
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,223 @@ | |||||||
|  | // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project | ||||||
|  | // SPDX-License-Identifier: GPL-2.0-or-later | ||||||
|  |  | ||||||
|  | #include <algorithm> | ||||||
|  | #include <numeric> | ||||||
|  | #include <vector> | ||||||
|  |  | ||||||
|  | #include "common/logging/log.h" | ||||||
|  | #include "common/settings.h" | ||||||
|  | #include "core/core.h" | ||||||
|  | #include "core/file_sys/common_funcs.h" | ||||||
|  | #include "core/file_sys/content_archive.h" | ||||||
|  | #include "core/file_sys/control_metadata.h" | ||||||
|  | #include "core/file_sys/nca_metadata.h" | ||||||
|  | #include "core/file_sys/patch_manager.h" | ||||||
|  | #include "core/file_sys/registered_cache.h" | ||||||
|  | #include "core/hle/kernel/k_event.h" | ||||||
|  | #include "core/hle/service/aoc/addon_content_manager.h" | ||||||
|  | #include "core/hle/service/aoc/purchase_event_manager.h" | ||||||
|  | #include "core/hle/service/cmif_serialization.h" | ||||||
|  | #include "core/hle/service/ipc_helpers.h" | ||||||
|  | #include "core/hle/service/server_manager.h" | ||||||
|  | #include "core/loader/loader.h" | ||||||
|  |  | ||||||
|  | namespace Service::AOC { | ||||||
|  |  | ||||||
|  | static bool CheckAOCTitleIDMatchesBase(u64 title_id, u64 base) { | ||||||
|  |     return FileSys::GetBaseTitleID(title_id) == base; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static std::vector<u64> AccumulateAOCTitleIDs(Core::System& system) { | ||||||
|  |     std::vector<u64> add_on_content; | ||||||
|  |     const auto& rcu = system.GetContentProvider(); | ||||||
|  |     const auto list = | ||||||
|  |         rcu.ListEntriesFilter(FileSys::TitleType::AOC, FileSys::ContentRecordType::Data); | ||||||
|  |     std::transform(list.begin(), list.end(), std::back_inserter(add_on_content), | ||||||
|  |                    [](const FileSys::ContentProviderEntry& rce) { return rce.title_id; }); | ||||||
|  |     add_on_content.erase( | ||||||
|  |         std::remove_if( | ||||||
|  |             add_on_content.begin(), add_on_content.end(), | ||||||
|  |             [&rcu](u64 tid) { | ||||||
|  |                 return rcu.GetEntry(tid, FileSys::ContentRecordType::Data)->GetStatus() != | ||||||
|  |                        Loader::ResultStatus::Success; | ||||||
|  |             }), | ||||||
|  |         add_on_content.end()); | ||||||
|  |     return add_on_content; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | IAddOnContentManager::IAddOnContentManager(Core::System& system_) | ||||||
|  |     : ServiceFramework{system_, "aoc:u"}, add_on_content{AccumulateAOCTitleIDs(system)}, | ||||||
|  |       service_context{system_, "aoc:u"} { | ||||||
|  |     // clang-format off | ||||||
|  |     static const FunctionInfo functions[] = { | ||||||
|  |         {0, nullptr, "CountAddOnContentByApplicationId"}, | ||||||
|  |         {1, nullptr, "ListAddOnContentByApplicationId"}, | ||||||
|  |         {2, D<&IAddOnContentManager::CountAddOnContent>, "CountAddOnContent"}, | ||||||
|  |         {3, D<&IAddOnContentManager::ListAddOnContent>, "ListAddOnContent"}, | ||||||
|  |         {4, nullptr, "GetAddOnContentBaseIdByApplicationId"}, | ||||||
|  |         {5, D<&IAddOnContentManager::GetAddOnContentBaseId>, "GetAddOnContentBaseId"}, | ||||||
|  |         {6, nullptr, "PrepareAddOnContentByApplicationId"}, | ||||||
|  |         {7, D<&IAddOnContentManager::PrepareAddOnContent>, "PrepareAddOnContent"}, | ||||||
|  |         {8, D<&IAddOnContentManager::GetAddOnContentListChangedEvent>, "GetAddOnContentListChangedEvent"}, | ||||||
|  |         {9, nullptr, "GetAddOnContentLostErrorCode"}, | ||||||
|  |         {10, D<&IAddOnContentManager::GetAddOnContentListChangedEventWithProcessId>, "GetAddOnContentListChangedEventWithProcessId"}, | ||||||
|  |         {11, D<&IAddOnContentManager::NotifyMountAddOnContent>, "NotifyMountAddOnContent"}, | ||||||
|  |         {12, D<&IAddOnContentManager::NotifyUnmountAddOnContent>, "NotifyUnmountAddOnContent"}, | ||||||
|  |         {13, nullptr, "IsAddOnContentMountedForDebug"}, | ||||||
|  |         {50, D<&IAddOnContentManager::CheckAddOnContentMountStatus>, "CheckAddOnContentMountStatus"}, | ||||||
|  |         {100, D<&IAddOnContentManager::CreateEcPurchasedEventManager>, "CreateEcPurchasedEventManager"}, | ||||||
|  |         {101, D<&IAddOnContentManager::CreatePermanentEcPurchasedEventManager>, "CreatePermanentEcPurchasedEventManager"}, | ||||||
|  |         {110, nullptr, "CreateContentsServiceManager"}, | ||||||
|  |         {200, nullptr, "SetRequiredAddOnContentsOnContentsAvailabilityTransition"}, | ||||||
|  |         {300, nullptr, "SetupHostAddOnContent"}, | ||||||
|  |         {301, nullptr, "GetRegisteredAddOnContentPath"}, | ||||||
|  |         {302, nullptr, "UpdateCachedList"}, | ||||||
|  |     }; | ||||||
|  |     // clang-format on | ||||||
|  |  | ||||||
|  |     RegisterHandlers(functions); | ||||||
|  |  | ||||||
|  |     aoc_change_event = service_context.CreateEvent("GetAddOnContentListChanged:Event"); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | IAddOnContentManager::~IAddOnContentManager() { | ||||||
|  |     service_context.CloseEvent(aoc_change_event); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IAddOnContentManager::CountAddOnContent(Out<u32> out_count, ClientProcessId process_id) { | ||||||
|  |     LOG_DEBUG(Service_AOC, "called. process_id={}", process_id.pid); | ||||||
|  |  | ||||||
|  |     const auto current = system.GetApplicationProcessProgramID(); | ||||||
|  |  | ||||||
|  |     const auto& disabled = Settings::values.disabled_addons[current]; | ||||||
|  |     if (std::find(disabled.begin(), disabled.end(), "DLC") != disabled.end()) { | ||||||
|  |         *out_count = 0; | ||||||
|  |         R_SUCCEED(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     *out_count = static_cast<u32>( | ||||||
|  |         std::count_if(add_on_content.begin(), add_on_content.end(), | ||||||
|  |                       [current](u64 tid) { return CheckAOCTitleIDMatchesBase(tid, current); })); | ||||||
|  |  | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IAddOnContentManager::ListAddOnContent(Out<u32> out_count, | ||||||
|  |                                               OutBuffer<BufferAttr_HipcMapAlias> out_addons, | ||||||
|  |                                               u32 offset, u32 count, ClientProcessId process_id) { | ||||||
|  |     LOG_DEBUG(Service_AOC, "called with offset={}, count={}, process_id={}", offset, count, | ||||||
|  |               process_id.pid); | ||||||
|  |  | ||||||
|  |     const auto current = FileSys::GetBaseTitleID(system.GetApplicationProcessProgramID()); | ||||||
|  |  | ||||||
|  |     std::vector<u32> out; | ||||||
|  |     const auto& disabled = Settings::values.disabled_addons[current]; | ||||||
|  |     if (std::find(disabled.begin(), disabled.end(), "DLC") == disabled.end()) { | ||||||
|  |         for (u64 content_id : add_on_content) { | ||||||
|  |             if (FileSys::GetBaseTitleID(content_id) != current) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             out.push_back(static_cast<u32>(FileSys::GetAOCID(content_id))); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // TODO(DarkLordZach): Find the correct error code. | ||||||
|  |     R_UNLESS(out.size() >= offset, ResultUnknown); | ||||||
|  |  | ||||||
|  |     *out_count = static_cast<u32>(std::min<size_t>(out.size() - offset, count)); | ||||||
|  |     std::rotate(out.begin(), out.begin() + offset, out.end()); | ||||||
|  |  | ||||||
|  |     std::memcpy(out_addons.data(), out.data(), *out_count * sizeof(u32)); | ||||||
|  |  | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IAddOnContentManager::GetAddOnContentBaseId(Out<u64> out_title_id, | ||||||
|  |                                                    ClientProcessId process_id) { | ||||||
|  |     LOG_DEBUG(Service_AOC, "called. process_id={}", process_id.pid); | ||||||
|  |  | ||||||
|  |     const auto title_id = system.GetApplicationProcessProgramID(); | ||||||
|  |     const FileSys::PatchManager pm{title_id, system.GetFileSystemController(), | ||||||
|  |                                    system.GetContentProvider()}; | ||||||
|  |  | ||||||
|  |     const auto res = pm.GetControlMetadata(); | ||||||
|  |     if (res.first == nullptr) { | ||||||
|  |         *out_title_id = FileSys::GetAOCBaseTitleID(title_id); | ||||||
|  |         R_SUCCEED(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     *out_title_id = res.first->GetDLCBaseTitleId(); | ||||||
|  |  | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IAddOnContentManager::PrepareAddOnContent(s32 addon_index, ClientProcessId process_id) { | ||||||
|  |     LOG_WARNING(Service_AOC, "(STUBBED) called with addon_index={}, process_id={}", addon_index, | ||||||
|  |                 process_id.pid); | ||||||
|  |  | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IAddOnContentManager::GetAddOnContentListChangedEvent( | ||||||
|  |     OutCopyHandle<Kernel::KReadableEvent> out_event) { | ||||||
|  |     LOG_WARNING(Service_AOC, "(STUBBED) called"); | ||||||
|  |  | ||||||
|  |     *out_event = &aoc_change_event->GetReadableEvent(); | ||||||
|  |  | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IAddOnContentManager::GetAddOnContentListChangedEventWithProcessId( | ||||||
|  |     OutCopyHandle<Kernel::KReadableEvent> out_event, ClientProcessId process_id) { | ||||||
|  |     LOG_WARNING(Service_AOC, "(STUBBED) called"); | ||||||
|  |  | ||||||
|  |     *out_event = &aoc_change_event->GetReadableEvent(); | ||||||
|  |  | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IAddOnContentManager::NotifyMountAddOnContent() { | ||||||
|  |     LOG_WARNING(Service_AOC, "(STUBBED) called"); | ||||||
|  |  | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IAddOnContentManager::NotifyUnmountAddOnContent() { | ||||||
|  |     LOG_WARNING(Service_AOC, "(STUBBED) called"); | ||||||
|  |  | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IAddOnContentManager::CheckAddOnContentMountStatus() { | ||||||
|  |     LOG_WARNING(Service_AOC, "(STUBBED) called"); | ||||||
|  |  | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IAddOnContentManager::CreateEcPurchasedEventManager( | ||||||
|  |     OutInterface<IPurchaseEventManager> out_interface) { | ||||||
|  |     LOG_WARNING(Service_AOC, "(STUBBED) called"); | ||||||
|  |  | ||||||
|  |     *out_interface = std::make_shared<IPurchaseEventManager>(system); | ||||||
|  |  | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IAddOnContentManager::CreatePermanentEcPurchasedEventManager( | ||||||
|  |     OutInterface<IPurchaseEventManager> out_interface) { | ||||||
|  |     LOG_WARNING(Service_AOC, "(STUBBED) called"); | ||||||
|  |  | ||||||
|  |     *out_interface = std::make_shared<IPurchaseEventManager>(system); | ||||||
|  |  | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void LoopProcess(Core::System& system) { | ||||||
|  |     auto server_manager = std::make_unique<ServerManager>(system); | ||||||
|  |     server_manager->RegisterNamedService("aoc:u", std::make_shared<IAddOnContentManager>(system)); | ||||||
|  |     ServerManager::RunServer(std::move(server_manager)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | } // namespace Service::AOC | ||||||
							
								
								
									
										51
									
								
								src/core/hle/service/aoc/addon_content_manager.h
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										51
									
								
								src/core/hle/service/aoc/addon_content_manager.h
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,51 @@ | |||||||
|  | // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project | ||||||
|  | // SPDX-License-Identifier: GPL-2.0-or-later | ||||||
|  |  | ||||||
|  | #pragma once | ||||||
|  |  | ||||||
|  | #include "core/hle/service/cmif_types.h" | ||||||
|  | #include "core/hle/service/kernel_helpers.h" | ||||||
|  | #include "core/hle/service/service.h" | ||||||
|  |  | ||||||
|  | namespace Core { | ||||||
|  | class System; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | namespace Kernel { | ||||||
|  | class KEvent; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | namespace Service::AOC { | ||||||
|  |  | ||||||
|  | class IPurchaseEventManager; | ||||||
|  |  | ||||||
|  | class IAddOnContentManager final : public ServiceFramework<IAddOnContentManager> { | ||||||
|  | public: | ||||||
|  |     explicit IAddOnContentManager(Core::System& system); | ||||||
|  |     ~IAddOnContentManager() override; | ||||||
|  |  | ||||||
|  |     Result CountAddOnContent(Out<u32> out_count, ClientProcessId process_id); | ||||||
|  |     Result ListAddOnContent(Out<u32> out_count, OutBuffer<BufferAttr_HipcMapAlias> out_addons, | ||||||
|  |                             u32 offset, u32 count, ClientProcessId process_id); | ||||||
|  |     Result GetAddOnContentBaseId(Out<u64> out_title_id, ClientProcessId process_id); | ||||||
|  |     Result PrepareAddOnContent(s32 addon_index, ClientProcessId process_id); | ||||||
|  |     Result GetAddOnContentListChangedEvent(OutCopyHandle<Kernel::KReadableEvent> out_event); | ||||||
|  |     Result GetAddOnContentListChangedEventWithProcessId( | ||||||
|  |         OutCopyHandle<Kernel::KReadableEvent> out_event, ClientProcessId process_id); | ||||||
|  |     Result NotifyMountAddOnContent(); | ||||||
|  |     Result NotifyUnmountAddOnContent(); | ||||||
|  |     Result CheckAddOnContentMountStatus(); | ||||||
|  |     Result CreateEcPurchasedEventManager(OutInterface<IPurchaseEventManager> out_interface); | ||||||
|  |     Result CreatePermanentEcPurchasedEventManager( | ||||||
|  |         OutInterface<IPurchaseEventManager> out_interface); | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     std::vector<u64> add_on_content; | ||||||
|  |     KernelHelpers::ServiceContext service_context; | ||||||
|  |  | ||||||
|  |     Kernel::KEvent* aoc_change_event; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | void LoopProcess(Core::System& system); | ||||||
|  |  | ||||||
|  | } // namespace Service::AOC | ||||||
							
								
								
									
										67
									
								
								src/core/hle/service/aoc/purchase_event_manager.cpp
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										67
									
								
								src/core/hle/service/aoc/purchase_event_manager.cpp
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,67 @@ | |||||||
|  | // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project | ||||||
|  | // SPDX-License-Identifier: GPL-2.0-or-later | ||||||
|  |  | ||||||
|  | #include "core/hle/service/aoc/purchase_event_manager.h" | ||||||
|  | #include "core/hle/service/cmif_serialization.h" | ||||||
|  |  | ||||||
|  | namespace Service::AOC { | ||||||
|  |  | ||||||
|  | constexpr Result ResultNoPurchasedProductInfoAvailable{ErrorModule::NIMShop, 400}; | ||||||
|  |  | ||||||
|  | IPurchaseEventManager::IPurchaseEventManager(Core::System& system_) | ||||||
|  |     : ServiceFramework{system_, "IPurchaseEventManager"}, service_context{system, | ||||||
|  |                                                                           "IPurchaseEventManager"} { | ||||||
|  |     // clang-format off | ||||||
|  |         static const FunctionInfo functions[] = { | ||||||
|  |             {0, D<&IPurchaseEventManager::SetDefaultDeliveryTarget>, "SetDefaultDeliveryTarget"}, | ||||||
|  |             {1, D<&IPurchaseEventManager::SetDeliveryTarget>, "SetDeliveryTarget"}, | ||||||
|  |             {2, D<&IPurchaseEventManager::GetPurchasedEvent>, "GetPurchasedEvent"}, | ||||||
|  |             {3, D<&IPurchaseEventManager::PopPurchasedProductInfo>, "PopPurchasedProductInfo"}, | ||||||
|  |             {4, D<&IPurchaseEventManager::PopPurchasedProductInfoWithUid>, "PopPurchasedProductInfoWithUid"}, | ||||||
|  |         }; | ||||||
|  |     // clang-format on | ||||||
|  |  | ||||||
|  |     RegisterHandlers(functions); | ||||||
|  |  | ||||||
|  |     purchased_event = service_context.CreateEvent("IPurchaseEventManager:PurchasedEvent"); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | IPurchaseEventManager::~IPurchaseEventManager() { | ||||||
|  |     service_context.CloseEvent(purchased_event); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IPurchaseEventManager::SetDefaultDeliveryTarget( | ||||||
|  |     ClientProcessId process_id, InBuffer<BufferAttr_HipcMapAlias> in_buffer) { | ||||||
|  |     LOG_WARNING(Service_AOC, "(STUBBED) called, process_id={}", process_id.pid); | ||||||
|  |  | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IPurchaseEventManager::SetDeliveryTarget(u64 unknown, | ||||||
|  |                                                 InBuffer<BufferAttr_HipcMapAlias> in_buffer) { | ||||||
|  |     LOG_WARNING(Service_AOC, "(STUBBED) called, unknown={}", unknown); | ||||||
|  |  | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IPurchaseEventManager::GetPurchasedEvent(OutCopyHandle<Kernel::KReadableEvent> out_event) { | ||||||
|  |     LOG_WARNING(Service_AOC, "called"); | ||||||
|  |  | ||||||
|  |     *out_event = &purchased_event->GetReadableEvent(); | ||||||
|  |  | ||||||
|  |     R_SUCCEED(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IPurchaseEventManager::PopPurchasedProductInfo() { | ||||||
|  |     LOG_DEBUG(Service_AOC, "(STUBBED) called"); | ||||||
|  |  | ||||||
|  |     R_RETURN(ResultNoPurchasedProductInfoAvailable); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Result IPurchaseEventManager::PopPurchasedProductInfoWithUid() { | ||||||
|  |     LOG_DEBUG(Service_AOC, "(STUBBED) called"); | ||||||
|  |  | ||||||
|  |     R_RETURN(ResultNoPurchasedProductInfoAvailable); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | } // namespace Service::AOC | ||||||
							
								
								
									
										30
									
								
								src/core/hle/service/aoc/purchase_event_manager.h
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										30
									
								
								src/core/hle/service/aoc/purchase_event_manager.h
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,30 @@ | |||||||
|  | // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project | ||||||
|  | // SPDX-License-Identifier: GPL-2.0-or-later | ||||||
|  |  | ||||||
|  | #pragma once | ||||||
|  |  | ||||||
|  | #include "core/hle/service/cmif_types.h" | ||||||
|  | #include "core/hle/service/kernel_helpers.h" | ||||||
|  | #include "core/hle/service/os/event.h" | ||||||
|  | #include "core/hle/service/service.h" | ||||||
|  |  | ||||||
|  | namespace Service::AOC { | ||||||
|  |  | ||||||
|  | class IPurchaseEventManager final : public ServiceFramework<IPurchaseEventManager> { | ||||||
|  | public: | ||||||
|  |     explicit IPurchaseEventManager(Core::System& system_); | ||||||
|  |     ~IPurchaseEventManager() override; | ||||||
|  |  | ||||||
|  |     Result SetDefaultDeliveryTarget(ClientProcessId process_id, | ||||||
|  |                                     InBuffer<BufferAttr_HipcMapAlias> in_buffer); | ||||||
|  |     Result SetDeliveryTarget(u64 unknown, InBuffer<BufferAttr_HipcMapAlias> in_buffer); | ||||||
|  |     Result GetPurchasedEvent(OutCopyHandle<Kernel::KReadableEvent> out_event); | ||||||
|  |     Result PopPurchasedProductInfo(); | ||||||
|  |     Result PopPurchasedProductInfoWithUid(); | ||||||
|  |  | ||||||
|  | private: | ||||||
|  |     KernelHelpers::ServiceContext service_context; | ||||||
|  |     Kernel::KEvent* purchased_event; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | } // namespace Service::AOC | ||||||
| @@ -5,7 +5,7 @@ | |||||||
|  |  | ||||||
| #include "core/hle/service/acc/acc.h" | #include "core/hle/service/acc/acc.h" | ||||||
| #include "core/hle/service/am/am.h" | #include "core/hle/service/am/am.h" | ||||||
| #include "core/hle/service/aoc/aoc_u.h" | #include "core/hle/service/aoc/addon_content_manager.h" | ||||||
| #include "core/hle/service/apm/apm.h" | #include "core/hle/service/apm/apm.h" | ||||||
| #include "core/hle/service/audio/audio.h" | #include "core/hle/service/audio/audio.h" | ||||||
| #include "core/hle/service/bcat/bcat.h" | #include "core/hle/service/bcat/bcat.h" | ||||||
|   | |||||||
| @@ -3010,9 +3010,6 @@ bool GMainWindow::MakeShortcutIcoPath(const u64 program_id, const std::string_vi | |||||||
|  |  | ||||||
| void GMainWindow::OnGameListCreateShortcut(u64 program_id, const std::string& game_path, | void GMainWindow::OnGameListCreateShortcut(u64 program_id, const std::string& game_path, | ||||||
|                                            GameListShortcutTarget target) { |                                            GameListShortcutTarget target) { | ||||||
|     std::string game_title; |  | ||||||
|     QString qt_game_title; |  | ||||||
|     std::filesystem::path out_icon_path; |  | ||||||
|     // Get path to yuzu executable |     // Get path to yuzu executable | ||||||
|     const QStringList args = QApplication::arguments(); |     const QStringList args = QApplication::arguments(); | ||||||
|     std::filesystem::path yuzu_command = args[0].toStdString(); |     std::filesystem::path yuzu_command = args[0].toStdString(); | ||||||
| @@ -3029,48 +3026,51 @@ void GMainWindow::OnGameListCreateShortcut(u64 program_id, const std::string& ga | |||||||
|         shortcut_path = |         shortcut_path = | ||||||
|             QStandardPaths::writableLocation(QStandardPaths::ApplicationsLocation).toStdString(); |             QStandardPaths::writableLocation(QStandardPaths::ApplicationsLocation).toStdString(); | ||||||
|     } |     } | ||||||
|     // Icon path and title |  | ||||||
|     if (std::filesystem::exists(shortcut_path)) { |     if (!std::filesystem::exists(shortcut_path)) { | ||||||
|         // Get title from game file |         GMainWindow::CreateShortcutMessagesGUI( | ||||||
|         const FileSys::PatchManager pm{program_id, system->GetFileSystemController(), |             this, GMainWindow::CREATE_SHORTCUT_MSGBOX_ERROR, | ||||||
|                                        system->GetContentProvider()}; |             QString::fromStdString(shortcut_path.generic_string())); | ||||||
|         const auto control = pm.GetControlMetadata(); |         LOG_ERROR(Frontend, "Invalid shortcut target {}", shortcut_path.generic_string()); | ||||||
|         const auto loader = |  | ||||||
|             Loader::GetLoader(*system, vfs->OpenFile(game_path, FileSys::OpenMode::Read)); |  | ||||||
|         game_title = fmt::format("{:016X}", program_id); |  | ||||||
|         if (control.first != nullptr) { |  | ||||||
|             game_title = control.first->GetApplicationName(); |  | ||||||
|         } else { |  | ||||||
|             loader->ReadTitle(game_title); |  | ||||||
|         } |  | ||||||
|         // Delete illegal characters from title |  | ||||||
|         const std::string illegal_chars = "<>:\"/\\|?*."; |  | ||||||
|         for (auto it = game_title.rbegin(); it != game_title.rend(); ++it) { |  | ||||||
|             if (illegal_chars.find(*it) != std::string::npos) { |  | ||||||
|                 game_title.erase(it.base() - 1); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         qt_game_title = QString::fromStdString(game_title); |  | ||||||
|         // Get icon from game file |  | ||||||
|         std::vector<u8> icon_image_file{}; |  | ||||||
|         if (control.second != nullptr) { |  | ||||||
|             icon_image_file = control.second->ReadAllBytes(); |  | ||||||
|         } else if (loader->ReadIcon(icon_image_file) != Loader::ResultStatus::Success) { |  | ||||||
|             LOG_WARNING(Frontend, "Could not read icon from {:s}", game_path); |  | ||||||
|         } |  | ||||||
|         QImage icon_data = |  | ||||||
|             QImage::fromData(icon_image_file.data(), static_cast<int>(icon_image_file.size())); |  | ||||||
|         if (GMainWindow::MakeShortcutIcoPath(program_id, game_title, out_icon_path)) { |  | ||||||
|             if (!SaveIconToFile(out_icon_path, icon_data)) { |  | ||||||
|                 LOG_ERROR(Frontend, "Could not write icon to file"); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } else { |  | ||||||
|         GMainWindow::CreateShortcutMessagesGUI(this, GMainWindow::CREATE_SHORTCUT_MSGBOX_ERROR, |  | ||||||
|                                                qt_game_title); |  | ||||||
|         LOG_ERROR(Frontend, "Invalid shortcut target"); |  | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     // Get title from game file | ||||||
|  |     const FileSys::PatchManager pm{program_id, system->GetFileSystemController(), | ||||||
|  |                                    system->GetContentProvider()}; | ||||||
|  |     const auto control = pm.GetControlMetadata(); | ||||||
|  |     const auto loader = | ||||||
|  |         Loader::GetLoader(*system, vfs->OpenFile(game_path, FileSys::OpenMode::Read)); | ||||||
|  |     std::string game_title = fmt::format("{:016X}", program_id); | ||||||
|  |     if (control.first != nullptr) { | ||||||
|  |         game_title = control.first->GetApplicationName(); | ||||||
|  |     } else { | ||||||
|  |         loader->ReadTitle(game_title); | ||||||
|  |     } | ||||||
|  |     // Delete illegal characters from title | ||||||
|  |     const std::string illegal_chars = "<>:\"/\\|?*."; | ||||||
|  |     for (auto it = game_title.rbegin(); it != game_title.rend(); ++it) { | ||||||
|  |         if (illegal_chars.find(*it) != std::string::npos) { | ||||||
|  |             game_title.erase(it.base() - 1); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     const QString qt_game_title = QString::fromStdString(game_title); | ||||||
|  |     // Get icon from game file | ||||||
|  |     std::vector<u8> icon_image_file{}; | ||||||
|  |     if (control.second != nullptr) { | ||||||
|  |         icon_image_file = control.second->ReadAllBytes(); | ||||||
|  |     } else if (loader->ReadIcon(icon_image_file) != Loader::ResultStatus::Success) { | ||||||
|  |         LOG_WARNING(Frontend, "Could not read icon from {:s}", game_path); | ||||||
|  |     } | ||||||
|  |     QImage icon_data = | ||||||
|  |         QImage::fromData(icon_image_file.data(), static_cast<int>(icon_image_file.size())); | ||||||
|  |     std::filesystem::path out_icon_path; | ||||||
|  |     if (GMainWindow::MakeShortcutIcoPath(program_id, game_title, out_icon_path)) { | ||||||
|  |         if (!SaveIconToFile(out_icon_path, icon_data)) { | ||||||
|  |             LOG_ERROR(Frontend, "Could not write icon to file"); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
| #if defined(__linux__) | #if defined(__linux__) | ||||||
|     // Special case for AppImages |     // Special case for AppImages | ||||||
|     // Warn once if we are making a shortcut to a volatile AppImage |     // Warn once if we are making a shortcut to a volatile AppImage | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user