Compare commits
9 Commits
android-26
...
android-26
Author | SHA1 | Date | |
---|---|---|---|
2051b3d2d6 | |||
921d800d66 | |||
6ab071dfb1 | |||
49f77aae14 | |||
08f4877288 | |||
3ebc0de12f | |||
7f9ad00b63 | |||
257b0ac59a | |||
e646ef45ac |
3
.github/workflows/verify.yml
vendored
3
.github/workflows/verify.yml
vendored
@ -81,7 +81,8 @@ jobs:
|
|||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: |
|
run: |
|
||||||
brew install autoconf automake boost ccache ffmpeg fmt glslang hidapi libtool libusb lz4 ninja nlohmann-json openssl pkg-config qt@5 sdl2 speexdsp zlib zlib zstd
|
# workaround for https://github.com/actions/setup-python/issues/577
|
||||||
|
brew install autoconf automake boost@1.83 ccache ffmpeg fmt glslang hidapi libtool libusb lz4 ninja nlohmann-json openssl pkg-config qt@5 sdl2 speexdsp zlib zlib zstd || brew link --overwrite python@3.12
|
||||||
- name: Build
|
- name: Build
|
||||||
run: |
|
run: |
|
||||||
mkdir build
|
mkdir build
|
||||||
|
@ -2,10 +2,12 @@
|
|||||||
|----|----|----|----|----|
|
|----|----|----|----|----|
|
||||||
| [12461](https://github.com/yuzu-emu/yuzu//pull/12461) | [`2831f5dc6`](https://github.com/yuzu-emu/yuzu//pull/12461/files) | Rework Nvdec and VIC to fix out-of-order videos, and speed up decoding. | [Kelebek1](https://github.com/Kelebek1/) | Yes |
|
| [12461](https://github.com/yuzu-emu/yuzu//pull/12461) | [`2831f5dc6`](https://github.com/yuzu-emu/yuzu//pull/12461/files) | Rework Nvdec and VIC to fix out-of-order videos, and speed up decoding. | [Kelebek1](https://github.com/Kelebek1/) | Yes |
|
||||||
| [12749](https://github.com/yuzu-emu/yuzu//pull/12749) | [`aad4b0d6f`](https://github.com/yuzu-emu/yuzu//pull/12749/files) | general: workarounds for SMMU syncing issues | [liamwhite](https://github.com/liamwhite/) | Yes |
|
| [12749](https://github.com/yuzu-emu/yuzu//pull/12749) | [`aad4b0d6f`](https://github.com/yuzu-emu/yuzu//pull/12749/files) | general: workarounds for SMMU syncing issues | [liamwhite](https://github.com/liamwhite/) | Yes |
|
||||||
| [13081](https://github.com/yuzu-emu/yuzu//pull/13081) | [`2786d34dd`](https://github.com/yuzu-emu/yuzu//pull/13081/files) | aoc: Migrate to use cmif serialization | [FearlessTobi](https://github.com/FearlessTobi/) | Yes |
|
| [13073](https://github.com/yuzu-emu/yuzu//pull/13073) | [`b5a17b501`](https://github.com/yuzu-emu/yuzu//pull/13073/files) | fsp: Migrate remaining interfaces to cmif serialization | [FearlessTobi](https://github.com/FearlessTobi/) | Yes |
|
||||||
| [13096](https://github.com/yuzu-emu/yuzu//pull/13096) | [`0a8759057`](https://github.com/yuzu-emu/yuzu//pull/13096/files) | texture_cache: use two-pass collection for costly load resources | [liamwhite](https://github.com/liamwhite/) | Yes |
|
| [13096](https://github.com/yuzu-emu/yuzu//pull/13096) | [`0a8759057`](https://github.com/yuzu-emu/yuzu//pull/13096/files) | texture_cache: use two-pass collection for costly load resources | [liamwhite](https://github.com/liamwhite/) | Yes |
|
||||||
|
| [13100](https://github.com/yuzu-emu/yuzu//pull/13100) | [`c04567fad`](https://github.com/yuzu-emu/yuzu//pull/13100/files) | audio: move to new ipc | [liamwhite](https://github.com/liamwhite/) | Yes |
|
||||||
|
| [13115](https://github.com/yuzu-emu/yuzu//pull/13115) | [`89c2fd3d2`](https://github.com/yuzu-emu/yuzu//pull/13115/files) | olsc, pctl: move to new ipc | [liamwhite](https://github.com/liamwhite/) | Yes |
|
||||||
| [13116](https://github.com/yuzu-emu/yuzu//pull/13116) | [`54ed837ad`](https://github.com/yuzu-emu/yuzu//pull/13116/files) | android: Flip AB/XY for 8Bitdo controllers | [t895](https://github.com/t895/) | Yes |
|
| [13116](https://github.com/yuzu-emu/yuzu//pull/13116) | [`54ed837ad`](https://github.com/yuzu-emu/yuzu//pull/13116/files) | android: Flip AB/XY for 8Bitdo controllers | [t895](https://github.com/t895/) | Yes |
|
||||||
| [13122](https://github.com/yuzu-emu/yuzu//pull/13122) | [`505b3e4a7`](https://github.com/yuzu-emu/yuzu//pull/13122/files) | vk_rasterizer: flip scissor y on lower left origin mode | [liamwhite](https://github.com/liamwhite/) | Yes |
|
| [13117](https://github.com/yuzu-emu/yuzu//pull/13117) | [`e85466c1a`](https://github.com/yuzu-emu/yuzu//pull/13117/files) | psc: stub overlay notification channel | [liamwhite](https://github.com/liamwhite/) | Yes |
|
||||||
|
|
||||||
|
|
||||||
End of merge log. You can find the original README.md below the break.
|
End of merge log. You can find the original README.md below the break.
|
||||||
|
@ -146,15 +146,11 @@ public:
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (released < tags.size()) {
|
|
||||||
tags[released] = tag;
|
|
||||||
}
|
|
||||||
|
|
||||||
released++;
|
|
||||||
|
|
||||||
if (released >= tags.size()) {
|
if (released >= tags.size()) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
tags[released++] = tag;
|
||||||
}
|
}
|
||||||
|
|
||||||
return released;
|
return released;
|
||||||
|
@ -67,13 +67,9 @@ public:
|
|||||||
oboe::AudioStreamBuilder builder;
|
oboe::AudioStreamBuilder builder;
|
||||||
|
|
||||||
const auto result = ConfigureBuilder(builder, direction)->openStream(temp_stream);
|
const auto result = ConfigureBuilder(builder, direction)->openStream(temp_stream);
|
||||||
if (result == oboe::Result::OK) {
|
ASSERT(result == oboe::Result::OK);
|
||||||
return temp_stream->getChannelCount() >= 6 ? 6 : 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
LOG_ERROR(Audio_Sink, "Failed to open {} stream. Using default channel count 2",
|
return temp_stream->getChannelCount() >= 6 ? 6 : 2;
|
||||||
direction == oboe::Direction::Output ? "output" : "input");
|
|
||||||
return 2;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
@ -486,10 +486,8 @@ 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/addon_content_manager.cpp
|
hle/service/aoc/aoc_u.cpp
|
||||||
hle/service/aoc/addon_content_manager.h
|
hle/service/aoc/aoc_u.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
|
||||||
@ -625,8 +623,6 @@ add_library(core STATIC
|
|||||||
hle/service/filesystem/fsp/fsp_srv.cpp
|
hle/service/filesystem/fsp/fsp_srv.cpp
|
||||||
hle/service/filesystem/fsp/fsp_srv.h
|
hle/service/filesystem/fsp/fsp_srv.h
|
||||||
hle/service/filesystem/fsp/fsp_types.h
|
hle/service/filesystem/fsp/fsp_types.h
|
||||||
hle/service/filesystem/fsp/save_data_transfer_prohibiter.cpp
|
|
||||||
hle/service/filesystem/fsp/save_data_transfer_prohibiter.h
|
|
||||||
hle/service/filesystem/romfs_controller.cpp
|
hle/service/filesystem/romfs_controller.cpp
|
||||||
hle/service/filesystem/romfs_controller.h
|
hle/service/filesystem/romfs_controller.h
|
||||||
hle/service/filesystem/save_data_controller.cpp
|
hle/service/filesystem/save_data_controller.cpp
|
||||||
|
@ -164,19 +164,6 @@ static_assert(sizeof(SaveDataExtraData) == 0x200, "SaveDataExtraData has invalid
|
|||||||
static_assert(std::is_trivially_copyable_v<SaveDataExtraData>,
|
static_assert(std::is_trivially_copyable_v<SaveDataExtraData>,
|
||||||
"Data type must be trivially copyable.");
|
"Data type must be trivially copyable.");
|
||||||
|
|
||||||
struct SaveDataFilter {
|
|
||||||
bool use_program_id;
|
|
||||||
bool use_save_data_type;
|
|
||||||
bool use_user_id;
|
|
||||||
bool use_save_data_id;
|
|
||||||
bool use_index;
|
|
||||||
SaveDataRank rank;
|
|
||||||
SaveDataAttribute attribute;
|
|
||||||
};
|
|
||||||
static_assert(sizeof(SaveDataFilter) == 0x48, "SaveDataFilter has invalid size.");
|
|
||||||
static_assert(std::is_trivially_copyable_v<SaveDataFilter>,
|
|
||||||
"Data type must be trivially copyable.");
|
|
||||||
|
|
||||||
struct HashSalt {
|
struct HashSalt {
|
||||||
static constexpr size_t Size = 32;
|
static constexpr size_t Size = 32;
|
||||||
|
|
||||||
|
@ -1,223 +0,0 @@
|
|||||||
// 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
|
|
@ -1,51 +0,0 @@
|
|||||||
// 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
|
|
340
src/core/hle/service/aoc/aoc_u.cpp
Normal file
340
src/core/hle/service/aoc/aoc_u.cpp
Normal file
@ -0,0 +1,340 @@
|
|||||||
|
// 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/aoc_u.h"
|
||||||
|
#include "core/hle/service/ipc_helpers.h"
|
||||||
|
#include "core/hle/service/server_manager.h"
|
||||||
|
#include "core/loader/loader.h"
|
||||||
|
|
||||||
|
namespace Service::AOC {
|
||||||
|
|
||||||
|
constexpr Result ResultNoPurchasedProductInfoAvailable{ErrorModule::NIMShop, 400};
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
class IPurchaseEventManager final : public ServiceFramework<IPurchaseEventManager> {
|
||||||
|
public:
|
||||||
|
explicit IPurchaseEventManager(Core::System& system_)
|
||||||
|
: ServiceFramework{system_, "IPurchaseEventManager"}, service_context{
|
||||||
|
system, "IPurchaseEventManager"} {
|
||||||
|
// clang-format off
|
||||||
|
static const FunctionInfo functions[] = {
|
||||||
|
{0, &IPurchaseEventManager::SetDefaultDeliveryTarget, "SetDefaultDeliveryTarget"},
|
||||||
|
{1, &IPurchaseEventManager::SetDeliveryTarget, "SetDeliveryTarget"},
|
||||||
|
{2, &IPurchaseEventManager::GetPurchasedEventReadableHandle, "GetPurchasedEventReadableHandle"},
|
||||||
|
{3, &IPurchaseEventManager::PopPurchasedProductInfo, "PopPurchasedProductInfo"},
|
||||||
|
{4, &IPurchaseEventManager::PopPurchasedProductInfoWithUid, "PopPurchasedProductInfoWithUid"},
|
||||||
|
};
|
||||||
|
// clang-format on
|
||||||
|
|
||||||
|
RegisterHandlers(functions);
|
||||||
|
|
||||||
|
purchased_event = service_context.CreateEvent("IPurchaseEventManager:PurchasedEvent");
|
||||||
|
}
|
||||||
|
|
||||||
|
~IPurchaseEventManager() override {
|
||||||
|
service_context.CloseEvent(purchased_event);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
void SetDefaultDeliveryTarget(HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
|
||||||
|
const auto unknown_1 = rp.Pop<u64>();
|
||||||
|
[[maybe_unused]] const auto unknown_2 = ctx.ReadBuffer();
|
||||||
|
|
||||||
|
LOG_WARNING(Service_AOC, "(STUBBED) called, unknown_1={}", unknown_1);
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetDeliveryTarget(HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
|
||||||
|
const auto unknown_1 = rp.Pop<u64>();
|
||||||
|
[[maybe_unused]] const auto unknown_2 = ctx.ReadBuffer();
|
||||||
|
|
||||||
|
LOG_WARNING(Service_AOC, "(STUBBED) called, unknown_1={}", unknown_1);
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
}
|
||||||
|
|
||||||
|
void GetPurchasedEventReadableHandle(HLERequestContext& ctx) {
|
||||||
|
LOG_WARNING(Service_AOC, "called");
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2, 1};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
rb.PushCopyObjects(purchased_event->GetReadableEvent());
|
||||||
|
}
|
||||||
|
|
||||||
|
void PopPurchasedProductInfo(HLERequestContext& ctx) {
|
||||||
|
LOG_DEBUG(Service_AOC, "(STUBBED) called");
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(ResultNoPurchasedProductInfoAvailable);
|
||||||
|
}
|
||||||
|
|
||||||
|
void PopPurchasedProductInfoWithUid(HLERequestContext& ctx) {
|
||||||
|
LOG_DEBUG(Service_AOC, "(STUBBED) called");
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(ResultNoPurchasedProductInfoAvailable);
|
||||||
|
}
|
||||||
|
|
||||||
|
KernelHelpers::ServiceContext service_context;
|
||||||
|
|
||||||
|
Kernel::KEvent* purchased_event;
|
||||||
|
};
|
||||||
|
|
||||||
|
AOC_U::AOC_U(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, &AOC_U::CountAddOnContent, "CountAddOnContent"},
|
||||||
|
{3, &AOC_U::ListAddOnContent, "ListAddOnContent"},
|
||||||
|
{4, nullptr, "GetAddOnContentBaseIdByApplicationId"},
|
||||||
|
{5, &AOC_U::GetAddOnContentBaseId, "GetAddOnContentBaseId"},
|
||||||
|
{6, nullptr, "PrepareAddOnContentByApplicationId"},
|
||||||
|
{7, &AOC_U::PrepareAddOnContent, "PrepareAddOnContent"},
|
||||||
|
{8, &AOC_U::GetAddOnContentListChangedEvent, "GetAddOnContentListChangedEvent"},
|
||||||
|
{9, nullptr, "GetAddOnContentLostErrorCode"},
|
||||||
|
{10, &AOC_U::GetAddOnContentListChangedEventWithProcessId, "GetAddOnContentListChangedEventWithProcessId"},
|
||||||
|
{11, &AOC_U::NotifyMountAddOnContent, "NotifyMountAddOnContent"},
|
||||||
|
{12, &AOC_U::NotifyUnmountAddOnContent, "NotifyUnmountAddOnContent"},
|
||||||
|
{13, nullptr, "IsAddOnContentMountedForDebug"},
|
||||||
|
{50, &AOC_U::CheckAddOnContentMountStatus, "CheckAddOnContentMountStatus"},
|
||||||
|
{100, &AOC_U::CreateEcPurchasedEventManager, "CreateEcPurchasedEventManager"},
|
||||||
|
{101, &AOC_U::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");
|
||||||
|
}
|
||||||
|
|
||||||
|
AOC_U::~AOC_U() {
|
||||||
|
service_context.CloseEvent(aoc_change_event);
|
||||||
|
}
|
||||||
|
|
||||||
|
void AOC_U::CountAddOnContent(HLERequestContext& ctx) {
|
||||||
|
struct Parameters {
|
||||||
|
u64 process_id;
|
||||||
|
};
|
||||||
|
static_assert(sizeof(Parameters) == 8);
|
||||||
|
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto params = rp.PopRaw<Parameters>();
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_AOC, "called. process_id={}", params.process_id);
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 3};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
|
||||||
|
const auto current = system.GetApplicationProcessProgramID();
|
||||||
|
|
||||||
|
const auto& disabled = Settings::values.disabled_addons[current];
|
||||||
|
if (std::find(disabled.begin(), disabled.end(), "DLC") != disabled.end()) {
|
||||||
|
rb.Push<u32>(0);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
rb.Push<u32>(static_cast<u32>(
|
||||||
|
std::count_if(add_on_content.begin(), add_on_content.end(),
|
||||||
|
[current](u64 tid) { return CheckAOCTitleIDMatchesBase(tid, current); })));
|
||||||
|
}
|
||||||
|
|
||||||
|
void AOC_U::ListAddOnContent(HLERequestContext& ctx) {
|
||||||
|
struct Parameters {
|
||||||
|
u32 offset;
|
||||||
|
u32 count;
|
||||||
|
u64 process_id;
|
||||||
|
};
|
||||||
|
static_assert(sizeof(Parameters) == 16);
|
||||||
|
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto [offset, count, process_id] = rp.PopRaw<Parameters>();
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_AOC, "called with offset={}, count={}, process_id={}", offset, count,
|
||||||
|
process_id);
|
||||||
|
|
||||||
|
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)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (out.size() < offset) {
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
// TODO(DarkLordZach): Find the correct error code.
|
||||||
|
rb.Push(ResultUnknown);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto out_count = static_cast<u32>(std::min<size_t>(out.size() - offset, count));
|
||||||
|
std::rotate(out.begin(), out.begin() + offset, out.end());
|
||||||
|
out.resize(out_count);
|
||||||
|
|
||||||
|
ctx.WriteBuffer(out);
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 3};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
rb.Push(out_count);
|
||||||
|
}
|
||||||
|
|
||||||
|
void AOC_U::GetAddOnContentBaseId(HLERequestContext& ctx) {
|
||||||
|
struct Parameters {
|
||||||
|
u64 process_id;
|
||||||
|
};
|
||||||
|
static_assert(sizeof(Parameters) == 8);
|
||||||
|
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto params = rp.PopRaw<Parameters>();
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_AOC, "called. process_id={}", params.process_id);
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 4};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
|
||||||
|
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) {
|
||||||
|
rb.Push(FileSys::GetAOCBaseTitleID(title_id));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
rb.Push(res.first->GetDLCBaseTitleId());
|
||||||
|
}
|
||||||
|
|
||||||
|
void AOC_U::PrepareAddOnContent(HLERequestContext& ctx) {
|
||||||
|
struct Parameters {
|
||||||
|
s32 addon_index;
|
||||||
|
u64 process_id;
|
||||||
|
};
|
||||||
|
static_assert(sizeof(Parameters) == 16);
|
||||||
|
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto [addon_index, process_id] = rp.PopRaw<Parameters>();
|
||||||
|
|
||||||
|
LOG_WARNING(Service_AOC, "(STUBBED) called with addon_index={}, process_id={}", addon_index,
|
||||||
|
process_id);
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
}
|
||||||
|
|
||||||
|
void AOC_U::GetAddOnContentListChangedEvent(HLERequestContext& ctx) {
|
||||||
|
LOG_WARNING(Service_AOC, "(STUBBED) called");
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2, 1};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
rb.PushCopyObjects(aoc_change_event->GetReadableEvent());
|
||||||
|
}
|
||||||
|
|
||||||
|
void AOC_U::GetAddOnContentListChangedEventWithProcessId(HLERequestContext& ctx) {
|
||||||
|
LOG_WARNING(Service_AOC, "(STUBBED) called");
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2, 1};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
rb.PushCopyObjects(aoc_change_event->GetReadableEvent());
|
||||||
|
}
|
||||||
|
|
||||||
|
void AOC_U::NotifyMountAddOnContent(HLERequestContext& ctx) {
|
||||||
|
LOG_WARNING(Service_AOC, "(STUBBED) called");
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
}
|
||||||
|
|
||||||
|
void AOC_U::NotifyUnmountAddOnContent(HLERequestContext& ctx) {
|
||||||
|
LOG_WARNING(Service_AOC, "(STUBBED) called");
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
}
|
||||||
|
|
||||||
|
void AOC_U::CheckAddOnContentMountStatus(HLERequestContext& ctx) {
|
||||||
|
LOG_WARNING(Service_AOC, "(STUBBED) called");
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
}
|
||||||
|
|
||||||
|
void AOC_U::CreateEcPurchasedEventManager(HLERequestContext& ctx) {
|
||||||
|
LOG_WARNING(Service_AOC, "(STUBBED) called");
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
rb.PushIpcInterface<IPurchaseEventManager>(system);
|
||||||
|
}
|
||||||
|
|
||||||
|
void AOC_U::CreatePermanentEcPurchasedEventManager(HLERequestContext& ctx) {
|
||||||
|
LOG_WARNING(Service_AOC, "(STUBBED) called");
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
||||||
|
rb.Push(ResultSuccess);
|
||||||
|
rb.PushIpcInterface<IPurchaseEventManager>(system);
|
||||||
|
}
|
||||||
|
|
||||||
|
void LoopProcess(Core::System& system) {
|
||||||
|
auto server_manager = std::make_unique<ServerManager>(system);
|
||||||
|
server_manager->RegisterNamedService("aoc:u", std::make_shared<AOC_U>(system));
|
||||||
|
ServerManager::RunServer(std::move(server_manager));
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Service::AOC
|
45
src/core/hle/service/aoc/aoc_u.h
Normal file
45
src/core/hle/service/aoc/aoc_u.h
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
// SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#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 AOC_U final : public ServiceFramework<AOC_U> {
|
||||||
|
public:
|
||||||
|
explicit AOC_U(Core::System& system);
|
||||||
|
~AOC_U() override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
void CountAddOnContent(HLERequestContext& ctx);
|
||||||
|
void ListAddOnContent(HLERequestContext& ctx);
|
||||||
|
void GetAddOnContentBaseId(HLERequestContext& ctx);
|
||||||
|
void PrepareAddOnContent(HLERequestContext& ctx);
|
||||||
|
void GetAddOnContentListChangedEvent(HLERequestContext& ctx);
|
||||||
|
void GetAddOnContentListChangedEventWithProcessId(HLERequestContext& ctx);
|
||||||
|
void NotifyMountAddOnContent(HLERequestContext& ctx);
|
||||||
|
void NotifyUnmountAddOnContent(HLERequestContext& ctx);
|
||||||
|
void CheckAddOnContentMountStatus(HLERequestContext& ctx);
|
||||||
|
void CreateEcPurchasedEventManager(HLERequestContext& ctx);
|
||||||
|
void CreatePermanentEcPurchasedEventManager(HLERequestContext& ctx);
|
||||||
|
|
||||||
|
std::vector<u64> add_on_content;
|
||||||
|
KernelHelpers::ServiceContext service_context;
|
||||||
|
|
||||||
|
Kernel::KEvent* aoc_change_event;
|
||||||
|
};
|
||||||
|
|
||||||
|
void LoopProcess(Core::System& system);
|
||||||
|
|
||||||
|
} // namespace Service::AOC
|
@ -1,67 +0,0 @@
|
|||||||
// 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
|
|
@ -1,30 +0,0 @@
|
|||||||
// 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
|
|
@ -34,7 +34,6 @@
|
|||||||
#include "core/hle/service/filesystem/fsp/fs_i_save_data_info_reader.h"
|
#include "core/hle/service/filesystem/fsp/fs_i_save_data_info_reader.h"
|
||||||
#include "core/hle/service/filesystem/fsp/fs_i_storage.h"
|
#include "core/hle/service/filesystem/fsp/fs_i_storage.h"
|
||||||
#include "core/hle/service/filesystem/fsp/fsp_srv.h"
|
#include "core/hle/service/filesystem/fsp/fsp_srv.h"
|
||||||
#include "core/hle/service/filesystem/fsp/save_data_transfer_prohibiter.h"
|
|
||||||
#include "core/hle/service/filesystem/romfs_controller.h"
|
#include "core/hle/service/filesystem/romfs_controller.h"
|
||||||
#include "core/hle/service/filesystem/save_data_controller.h"
|
#include "core/hle/service/filesystem/save_data_controller.h"
|
||||||
#include "core/hle/service/hle_ipc.h"
|
#include "core/hle/service/hle_ipc.h"
|
||||||
@ -88,7 +87,7 @@ FSP_SRV::FSP_SRV(Core::System& system_)
|
|||||||
{64, nullptr, "OpenSaveDataInternalStorageFileSystem"},
|
{64, nullptr, "OpenSaveDataInternalStorageFileSystem"},
|
||||||
{65, nullptr, "UpdateSaveDataMacForDebug"},
|
{65, nullptr, "UpdateSaveDataMacForDebug"},
|
||||||
{66, nullptr, "WriteSaveDataFileSystemExtraData2"},
|
{66, nullptr, "WriteSaveDataFileSystemExtraData2"},
|
||||||
{67, D<&FSP_SRV::FindSaveDataWithFilter>, "FindSaveDataWithFilter"},
|
{67, nullptr, "FindSaveDataWithFilter"},
|
||||||
{68, nullptr, "OpenSaveDataInfoReaderBySaveDataFilter"},
|
{68, nullptr, "OpenSaveDataInfoReaderBySaveDataFilter"},
|
||||||
{69, nullptr, "ReadSaveDataFileSystemExtraDataBySaveDataAttribute"},
|
{69, nullptr, "ReadSaveDataFileSystemExtraDataBySaveDataAttribute"},
|
||||||
{70, D<&FSP_SRV::WriteSaveDataFileSystemExtraDataBySaveDataAttribute>, "WriteSaveDataFileSystemExtraDataBySaveDataAttribute"},
|
{70, D<&FSP_SRV::WriteSaveDataFileSystemExtraDataBySaveDataAttribute>, "WriteSaveDataFileSystemExtraDataBySaveDataAttribute"},
|
||||||
@ -96,7 +95,7 @@ FSP_SRV::FSP_SRV(Core::System& system_)
|
|||||||
{80, nullptr, "OpenSaveDataMetaFile"},
|
{80, nullptr, "OpenSaveDataMetaFile"},
|
||||||
{81, nullptr, "OpenSaveDataTransferManager"},
|
{81, nullptr, "OpenSaveDataTransferManager"},
|
||||||
{82, nullptr, "OpenSaveDataTransferManagerVersion2"},
|
{82, nullptr, "OpenSaveDataTransferManagerVersion2"},
|
||||||
{83, D<&FSP_SRV::OpenSaveDataTransferProhibiter>, "OpenSaveDataTransferProhibiter"},
|
{83, nullptr, "OpenSaveDataTransferProhibiterForCloudBackUp"},
|
||||||
{84, nullptr, "ListApplicationAccessibleSaveDataOwnerId"},
|
{84, nullptr, "ListApplicationAccessibleSaveDataOwnerId"},
|
||||||
{85, nullptr, "OpenSaveDataTransferManagerForSaveDataRepair"},
|
{85, nullptr, "OpenSaveDataTransferManagerForSaveDataRepair"},
|
||||||
{86, nullptr, "OpenSaveDataMover"},
|
{86, nullptr, "OpenSaveDataMover"},
|
||||||
@ -236,7 +235,7 @@ Result FSP_SRV::CreateSaveDataFileSystem(FileSys::SaveDataCreationInfo save_crea
|
|||||||
}
|
}
|
||||||
|
|
||||||
Result FSP_SRV::CreateSaveDataFileSystemBySystemSaveDataId(
|
Result FSP_SRV::CreateSaveDataFileSystemBySystemSaveDataId(
|
||||||
FileSys::SaveDataAttribute save_struct, FileSys::SaveDataCreationInfo save_create_struct) {
|
FileSys::SaveDataCreationInfo save_create_struct, FileSys::SaveDataAttribute save_struct) {
|
||||||
LOG_DEBUG(Service_FS, "called save_struct = {}", save_struct.DebugInfo());
|
LOG_DEBUG(Service_FS, "called save_struct = {}", save_struct.DebugInfo());
|
||||||
|
|
||||||
FileSys::VirtualDir save_data_dir{};
|
FileSys::VirtualDir save_data_dir{};
|
||||||
@ -309,14 +308,6 @@ Result FSP_SRV::OpenSaveDataInfoReaderOnlyCacheStorage(
|
|||||||
R_SUCCEED();
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
Result FSP_SRV::FindSaveDataWithFilter(Out<s64> out_count,
|
|
||||||
OutBuffer<BufferAttr_HipcMapAlias> out_buffer,
|
|
||||||
FileSys::SaveDataSpaceId space_id,
|
|
||||||
FileSys::SaveDataFilter filter) {
|
|
||||||
LOG_WARNING(Service_FS, "(STUBBED) called");
|
|
||||||
R_THROW(FileSys::ResultTargetNotFound);
|
|
||||||
}
|
|
||||||
|
|
||||||
Result FSP_SRV::WriteSaveDataFileSystemExtraDataBySaveDataAttribute() {
|
Result FSP_SRV::WriteSaveDataFileSystemExtraDataBySaveDataAttribute() {
|
||||||
LOG_WARNING(Service_FS, "(STUBBED) called.");
|
LOG_WARNING(Service_FS, "(STUBBED) called.");
|
||||||
|
|
||||||
@ -341,13 +332,6 @@ Result FSP_SRV::ReadSaveDataFileSystemExtraDataWithMaskBySaveDataAttribute(
|
|||||||
R_SUCCEED();
|
R_SUCCEED();
|
||||||
}
|
}
|
||||||
|
|
||||||
Result FSP_SRV::OpenSaveDataTransferProhibiter(
|
|
||||||
OutInterface<ISaveDataTransferProhibiter> out_prohibiter, u64 id) {
|
|
||||||
LOG_WARNING(Service_FS, "(STUBBED) called, id={:016X}", id);
|
|
||||||
*out_prohibiter = std::make_shared<ISaveDataTransferProhibiter>(system);
|
|
||||||
R_SUCCEED();
|
|
||||||
}
|
|
||||||
|
|
||||||
Result FSP_SRV::OpenDataStorageByCurrentProcess(OutInterface<IStorage> out_interface) {
|
Result FSP_SRV::OpenDataStorageByCurrentProcess(OutInterface<IStorage> out_interface) {
|
||||||
LOG_DEBUG(Service_FS, "called");
|
LOG_DEBUG(Service_FS, "called");
|
||||||
|
|
||||||
|
@ -25,7 +25,6 @@ class SaveDataController;
|
|||||||
|
|
||||||
class IFileSystem;
|
class IFileSystem;
|
||||||
class ISaveDataInfoReader;
|
class ISaveDataInfoReader;
|
||||||
class ISaveDataTransferProhibiter;
|
|
||||||
class IStorage;
|
class IStorage;
|
||||||
class IMultiCommitManager;
|
class IMultiCommitManager;
|
||||||
|
|
||||||
@ -54,7 +53,7 @@ private:
|
|||||||
Result CreateSaveDataFileSystem(FileSys::SaveDataCreationInfo save_create_struct,
|
Result CreateSaveDataFileSystem(FileSys::SaveDataCreationInfo save_create_struct,
|
||||||
FileSys::SaveDataAttribute save_struct, u128 uid);
|
FileSys::SaveDataAttribute save_struct, u128 uid);
|
||||||
Result CreateSaveDataFileSystemBySystemSaveDataId(
|
Result CreateSaveDataFileSystemBySystemSaveDataId(
|
||||||
FileSys::SaveDataAttribute save_struct, FileSys::SaveDataCreationInfo save_create_struct);
|
FileSys::SaveDataCreationInfo save_create_struct, FileSys::SaveDataAttribute save_struct);
|
||||||
Result OpenSaveDataFileSystem(OutInterface<IFileSystem> out_interface,
|
Result OpenSaveDataFileSystem(OutInterface<IFileSystem> out_interface,
|
||||||
FileSys::SaveDataSpaceId space_id,
|
FileSys::SaveDataSpaceId space_id,
|
||||||
FileSys::SaveDataAttribute attribute);
|
FileSys::SaveDataAttribute attribute);
|
||||||
@ -67,16 +66,11 @@ private:
|
|||||||
Result OpenSaveDataInfoReaderBySaveDataSpaceId(OutInterface<ISaveDataInfoReader> out_interface,
|
Result OpenSaveDataInfoReaderBySaveDataSpaceId(OutInterface<ISaveDataInfoReader> out_interface,
|
||||||
FileSys::SaveDataSpaceId space);
|
FileSys::SaveDataSpaceId space);
|
||||||
Result OpenSaveDataInfoReaderOnlyCacheStorage(OutInterface<ISaveDataInfoReader> out_interface);
|
Result OpenSaveDataInfoReaderOnlyCacheStorage(OutInterface<ISaveDataInfoReader> out_interface);
|
||||||
Result FindSaveDataWithFilter(Out<s64> out_count, OutBuffer<BufferAttr_HipcMapAlias> out_buffer,
|
|
||||||
FileSys::SaveDataSpaceId space_id,
|
|
||||||
FileSys::SaveDataFilter filter);
|
|
||||||
Result WriteSaveDataFileSystemExtraDataBySaveDataAttribute();
|
Result WriteSaveDataFileSystemExtraDataBySaveDataAttribute();
|
||||||
Result ReadSaveDataFileSystemExtraDataWithMaskBySaveDataAttribute(
|
Result ReadSaveDataFileSystemExtraDataWithMaskBySaveDataAttribute(
|
||||||
FileSys::SaveDataSpaceId space_id, FileSys::SaveDataAttribute attribute,
|
FileSys::SaveDataSpaceId space_id, FileSys::SaveDataAttribute attribute,
|
||||||
InBuffer<BufferAttr_HipcMapAlias> mask_buffer,
|
InBuffer<BufferAttr_HipcMapAlias> mask_buffer,
|
||||||
OutBuffer<BufferAttr_HipcMapAlias> out_buffer);
|
OutBuffer<BufferAttr_HipcMapAlias> out_buffer);
|
||||||
Result OpenSaveDataTransferProhibiter(OutInterface<ISaveDataTransferProhibiter> out_prohibiter,
|
|
||||||
u64 id);
|
|
||||||
Result OpenDataStorageByCurrentProcess(OutInterface<IStorage> out_interface);
|
Result OpenDataStorageByCurrentProcess(OutInterface<IStorage> out_interface);
|
||||||
Result OpenDataStorageByDataId(OutInterface<IStorage> out_interface,
|
Result OpenDataStorageByDataId(OutInterface<IStorage> out_interface,
|
||||||
FileSys::StorageId storage_id, u32 unknown, u64 title_id);
|
FileSys::StorageId storage_id, u32 unknown, u64 title_id);
|
||||||
|
@ -1,13 +0,0 @@
|
|||||||
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#include "core/hle/service/filesystem/fsp/save_data_transfer_prohibiter.h"
|
|
||||||
|
|
||||||
namespace Service::FileSystem {
|
|
||||||
|
|
||||||
ISaveDataTransferProhibiter::ISaveDataTransferProhibiter(Core::System& system_)
|
|
||||||
: ServiceFramework{system_, "ISaveDataTransferProhibiter"} {}
|
|
||||||
|
|
||||||
ISaveDataTransferProhibiter::~ISaveDataTransferProhibiter() = default;
|
|
||||||
|
|
||||||
} // namespace Service::FileSystem
|
|
@ -1,16 +0,0 @@
|
|||||||
// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project
|
|
||||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "core/hle/service/service.h"
|
|
||||||
|
|
||||||
namespace Service::FileSystem {
|
|
||||||
|
|
||||||
class ISaveDataTransferProhibiter : public ServiceFramework<ISaveDataTransferProhibiter> {
|
|
||||||
public:
|
|
||||||
explicit ISaveDataTransferProhibiter(Core::System& system_);
|
|
||||||
~ISaveDataTransferProhibiter() override;
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace Service::FileSystem
|
|
@ -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/addon_content_manager.h"
|
#include "core/hle/service/aoc/aoc_u.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"
|
||||||
|
@ -10,8 +10,6 @@
|
|||||||
#include "common/polyfill_thread.h"
|
#include "common/polyfill_thread.h"
|
||||||
#include "common/thread.h"
|
#include "common/thread.h"
|
||||||
|
|
||||||
#include "core/hle/service/vi/vsync_manager.h"
|
|
||||||
|
|
||||||
namespace Core {
|
namespace Core {
|
||||||
class System;
|
class System;
|
||||||
}
|
}
|
||||||
@ -28,6 +26,7 @@ namespace Service::VI {
|
|||||||
|
|
||||||
class Container;
|
class Container;
|
||||||
class DisplayList;
|
class DisplayList;
|
||||||
|
class VsyncManager;
|
||||||
|
|
||||||
class Conductor {
|
class Conductor {
|
||||||
public:
|
public:
|
||||||
|
@ -125,11 +125,9 @@ VkRect2D GetScissorState(const Maxwell& regs, size_t index, u32 up_scale = 1, u3
|
|||||||
return value < 0 ? std::min<s32>(converted_value - acumm, -1)
|
return value < 0 ? std::min<s32>(converted_value - acumm, -1)
|
||||||
: std::max<s32>(converted_value + acumm, 1);
|
: std::max<s32>(converted_value + acumm, 1);
|
||||||
};
|
};
|
||||||
const bool lower_left = regs.window_origin.mode != Maxwell::WindowOrigin::Mode::UpperLeft;
|
|
||||||
const s32 y_adj = lower_left ? scale_up(regs.surface_clip.height - (src.max_y - src.min_y)) : 0;
|
|
||||||
if (src.enable) {
|
if (src.enable) {
|
||||||
scissor.offset.x = scale_up(static_cast<s32>(src.min_x));
|
scissor.offset.x = scale_up(static_cast<s32>(src.min_x));
|
||||||
scissor.offset.y = scale_up(static_cast<s32>(src.min_y)) + y_adj;
|
scissor.offset.y = scale_up(static_cast<s32>(src.min_y));
|
||||||
scissor.extent.width = scale_up(src.max_x - src.min_x);
|
scissor.extent.width = scale_up(src.max_x - src.min_x);
|
||||||
scissor.extent.height = scale_up(src.max_y - src.min_y);
|
scissor.extent.height = scale_up(src.max_y - src.min_y);
|
||||||
} else {
|
} else {
|
||||||
|
@ -3010,6 +3010,9 @@ 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();
|
||||||
@ -3026,51 +3029,48 @@ 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)) {
|
||||||
GMainWindow::CreateShortcutMessagesGUI(
|
// Get title from game file
|
||||||
this, GMainWindow::CREATE_SHORTCUT_MSGBOX_ERROR,
|
const FileSys::PatchManager pm{program_id, system->GetFileSystemController(),
|
||||||
QString::fromStdString(shortcut_path.generic_string()));
|
system->GetContentProvider()};
|
||||||
LOG_ERROR(Frontend, "Invalid shortcut target {}", shortcut_path.generic_string());
|
const auto control = pm.GetControlMetadata();
|
||||||
|
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