diff options
Diffstat (limited to 'src')
86 files changed, 3903 insertions, 247 deletions
diff --git a/src/common/common_funcs.h b/src/common/common_funcs.h index 98421bced..367b6bf6e 100644 --- a/src/common/common_funcs.h +++ b/src/common/common_funcs.h | |||
| @@ -64,14 +64,20 @@ __declspec(dllimport) void __stdcall DebugBreak(void); | |||
| 64 | using T = std::underlying_type_t<type>; \ | 64 | using T = std::underlying_type_t<type>; \ |
| 65 | return static_cast<type>(static_cast<T>(a) & static_cast<T>(b)); \ | 65 | return static_cast<type>(static_cast<T>(a) & static_cast<T>(b)); \ |
| 66 | } \ | 66 | } \ |
| 67 | constexpr type& operator|=(type& a, type b) noexcept { \ | 67 | [[nodiscard]] constexpr type operator^(type a, type b) noexcept { \ |
| 68 | using T = std::underlying_type_t<type>; \ | 68 | using T = std::underlying_type_t<type>; \ |
| 69 | a = static_cast<type>(static_cast<T>(a) | static_cast<T>(b)); \ | 69 | return static_cast<type>(static_cast<T>(a) ^ static_cast<T>(b)); \ |
| 70 | } \ | ||
| 71 | constexpr type& operator|=(type& a, type b) noexcept { \ | ||
| 72 | a = a | b; \ | ||
| 70 | return a; \ | 73 | return a; \ |
| 71 | } \ | 74 | } \ |
| 72 | constexpr type& operator&=(type& a, type b) noexcept { \ | 75 | constexpr type& operator&=(type& a, type b) noexcept { \ |
| 73 | using T = std::underlying_type_t<type>; \ | 76 | a = a & b; \ |
| 74 | a = static_cast<type>(static_cast<T>(a) & static_cast<T>(b)); \ | 77 | return a; \ |
| 78 | } \ | ||
| 79 | constexpr type& operator^=(type& a, type b) noexcept { \ | ||
| 80 | a = a ^ b; \ | ||
| 75 | return a; \ | 81 | return a; \ |
| 76 | } \ | 82 | } \ |
| 77 | [[nodiscard]] constexpr type operator~(type key) noexcept { \ | 83 | [[nodiscard]] constexpr type operator~(type key) noexcept { \ |
diff --git a/src/common/math_util.h b/src/common/math_util.h index cc35c90ee..b35ad8507 100644 --- a/src/common/math_util.h +++ b/src/common/math_util.h | |||
| @@ -9,7 +9,7 @@ | |||
| 9 | 9 | ||
| 10 | namespace Common { | 10 | namespace Common { |
| 11 | 11 | ||
| 12 | constexpr float PI = 3.14159265f; | 12 | constexpr float PI = 3.1415926535f; |
| 13 | 13 | ||
| 14 | template <class T> | 14 | template <class T> |
| 15 | struct Rectangle { | 15 | struct Rectangle { |
diff --git a/src/common/quaternion.h b/src/common/quaternion.h index da44f35cd..4d0871eb4 100644 --- a/src/common/quaternion.h +++ b/src/common/quaternion.h | |||
| @@ -36,6 +36,36 @@ public: | |||
| 36 | T length = std::sqrt(xyz.Length2() + w * w); | 36 | T length = std::sqrt(xyz.Length2() + w * w); |
| 37 | return {xyz / length, w / length}; | 37 | return {xyz / length, w / length}; |
| 38 | } | 38 | } |
| 39 | |||
| 40 | [[nodiscard]] std::array<decltype(-T{}), 16> ToMatrix() const { | ||
| 41 | const T x2 = xyz[0] * xyz[0]; | ||
| 42 | const T y2 = xyz[1] * xyz[1]; | ||
| 43 | const T z2 = xyz[2] * xyz[2]; | ||
| 44 | |||
| 45 | const T xy = xyz[0] * xyz[1]; | ||
| 46 | const T wz = w * xyz[2]; | ||
| 47 | const T xz = xyz[0] * xyz[2]; | ||
| 48 | const T wy = w * xyz[1]; | ||
| 49 | const T yz = xyz[1] * xyz[2]; | ||
| 50 | const T wx = w * xyz[0]; | ||
| 51 | |||
| 52 | return {1.0f - 2.0f * (y2 + z2), | ||
| 53 | 2.0f * (xy + wz), | ||
| 54 | 2.0f * (xz - wy), | ||
| 55 | 0.0f, | ||
| 56 | 2.0f * (xy - wz), | ||
| 57 | 1.0f - 2.0f * (x2 + z2), | ||
| 58 | 2.0f * (yz + wx), | ||
| 59 | 0.0f, | ||
| 60 | 2.0f * (xz + wy), | ||
| 61 | 2.0f * (yz - wx), | ||
| 62 | 1.0f - 2.0f * (x2 + y2), | ||
| 63 | 0.0f, | ||
| 64 | 0.0f, | ||
| 65 | 0.0f, | ||
| 66 | 0.0f, | ||
| 67 | 1.0f}; | ||
| 68 | } | ||
| 39 | }; | 69 | }; |
| 40 | 70 | ||
| 41 | template <typename T> | 71 | template <typename T> |
diff --git a/src/common/thread.cpp b/src/common/thread.cpp index 8e5935e6a..d2c1ac60d 100644 --- a/src/common/thread.cpp +++ b/src/common/thread.cpp | |||
| @@ -2,6 +2,8 @@ | |||
| 2 | // Licensed under GPLv2 or any later version | 2 | // Licensed under GPLv2 or any later version |
| 3 | // Refer to the license.txt file included. | 3 | // Refer to the license.txt file included. |
| 4 | 4 | ||
| 5 | #include "common/common_funcs.h" | ||
| 6 | #include "common/logging/log.h" | ||
| 5 | #include "common/thread.h" | 7 | #include "common/thread.h" |
| 6 | #ifdef __APPLE__ | 8 | #ifdef __APPLE__ |
| 7 | #include <mach/mach.h> | 9 | #include <mach/mach.h> |
| @@ -19,6 +21,8 @@ | |||
| 19 | #include <unistd.h> | 21 | #include <unistd.h> |
| 20 | #endif | 22 | #endif |
| 21 | 23 | ||
| 24 | #include <string> | ||
| 25 | |||
| 22 | #ifdef __FreeBSD__ | 26 | #ifdef __FreeBSD__ |
| 23 | #define cpu_set_t cpuset_t | 27 | #define cpu_set_t cpuset_t |
| 24 | #endif | 28 | #endif |
| @@ -110,6 +114,14 @@ void SetCurrentThreadName(const char* name) { | |||
| 110 | pthread_set_name_np(pthread_self(), name); | 114 | pthread_set_name_np(pthread_self(), name); |
| 111 | #elif defined(__NetBSD__) | 115 | #elif defined(__NetBSD__) |
| 112 | pthread_setname_np(pthread_self(), "%s", (void*)name); | 116 | pthread_setname_np(pthread_self(), "%s", (void*)name); |
| 117 | #elif defined(__linux__) | ||
| 118 | // Linux limits thread names to 15 characters and will outright reject any | ||
| 119 | // attempt to set a longer name with ERANGE. | ||
| 120 | std::string truncated(name, std::min(strlen(name), static_cast<size_t>(15))); | ||
| 121 | if (int e = pthread_setname_np(pthread_self(), truncated.c_str())) { | ||
| 122 | errno = e; | ||
| 123 | LOG_ERROR(Common, "Failed to set thread name to '{}': {}", truncated, GetLastErrorMsg()); | ||
| 124 | } | ||
| 113 | #else | 125 | #else |
| 114 | pthread_setname_np(pthread_self(), name); | 126 | pthread_setname_np(pthread_self(), name); |
| 115 | #endif | 127 | #endif |
diff --git a/src/common/thread.h b/src/common/thread.h index 52b359413..a8c17c71a 100644 --- a/src/common/thread.h +++ b/src/common/thread.h | |||
| @@ -4,6 +4,7 @@ | |||
| 4 | 4 | ||
| 5 | #pragma once | 5 | #pragma once |
| 6 | 6 | ||
| 7 | #include <atomic> | ||
| 7 | #include <chrono> | 8 | #include <chrono> |
| 8 | #include <condition_variable> | 9 | #include <condition_variable> |
| 9 | #include <cstddef> | 10 | #include <cstddef> |
| @@ -25,13 +26,13 @@ public: | |||
| 25 | 26 | ||
| 26 | void Wait() { | 27 | void Wait() { |
| 27 | std::unique_lock lk{mutex}; | 28 | std::unique_lock lk{mutex}; |
| 28 | condvar.wait(lk, [&] { return is_set; }); | 29 | condvar.wait(lk, [&] { return is_set.load(); }); |
| 29 | is_set = false; | 30 | is_set = false; |
| 30 | } | 31 | } |
| 31 | 32 | ||
| 32 | bool WaitFor(const std::chrono::nanoseconds& time) { | 33 | bool WaitFor(const std::chrono::nanoseconds& time) { |
| 33 | std::unique_lock lk{mutex}; | 34 | std::unique_lock lk{mutex}; |
| 34 | if (!condvar.wait_for(lk, time, [this] { return is_set; })) | 35 | if (!condvar.wait_for(lk, time, [this] { return is_set.load(); })) |
| 35 | return false; | 36 | return false; |
| 36 | is_set = false; | 37 | is_set = false; |
| 37 | return true; | 38 | return true; |
| @@ -40,7 +41,7 @@ public: | |||
| 40 | template <class Clock, class Duration> | 41 | template <class Clock, class Duration> |
| 41 | bool WaitUntil(const std::chrono::time_point<Clock, Duration>& time) { | 42 | bool WaitUntil(const std::chrono::time_point<Clock, Duration>& time) { |
| 42 | std::unique_lock lk{mutex}; | 43 | std::unique_lock lk{mutex}; |
| 43 | if (!condvar.wait_until(lk, time, [this] { return is_set; })) | 44 | if (!condvar.wait_until(lk, time, [this] { return is_set.load(); })) |
| 44 | return false; | 45 | return false; |
| 45 | is_set = false; | 46 | is_set = false; |
| 46 | return true; | 47 | return true; |
| @@ -54,9 +55,9 @@ public: | |||
| 54 | } | 55 | } |
| 55 | 56 | ||
| 56 | private: | 57 | private: |
| 57 | bool is_set = false; | ||
| 58 | std::condition_variable condvar; | 58 | std::condition_variable condvar; |
| 59 | std::mutex mutex; | 59 | std::mutex mutex; |
| 60 | std::atomic_bool is_set{false}; | ||
| 60 | }; | 61 | }; |
| 61 | 62 | ||
| 62 | class Barrier { | 63 | class Barrier { |
diff --git a/src/common/x64/xbyak_abi.h b/src/common/x64/xbyak_abi.h index a5f5d4fc1..26e4bfda5 100644 --- a/src/common/x64/xbyak_abi.h +++ b/src/common/x64/xbyak_abi.h | |||
| @@ -11,7 +11,7 @@ | |||
| 11 | 11 | ||
| 12 | namespace Common::X64 { | 12 | namespace Common::X64 { |
| 13 | 13 | ||
| 14 | inline std::size_t RegToIndex(const Xbyak::Reg& reg) { | 14 | constexpr std::size_t RegToIndex(const Xbyak::Reg& reg) { |
| 15 | using Kind = Xbyak::Reg::Kind; | 15 | using Kind = Xbyak::Reg::Kind; |
| 16 | ASSERT_MSG((reg.getKind() & (Kind::REG | Kind::XMM)) != 0, | 16 | ASSERT_MSG((reg.getKind() & (Kind::REG | Kind::XMM)) != 0, |
| 17 | "RegSet only support GPRs and XMM registers."); | 17 | "RegSet only support GPRs and XMM registers."); |
| @@ -19,17 +19,17 @@ inline std::size_t RegToIndex(const Xbyak::Reg& reg) { | |||
| 19 | return reg.getIdx() + (reg.getKind() == Kind::REG ? 0 : 16); | 19 | return reg.getIdx() + (reg.getKind() == Kind::REG ? 0 : 16); |
| 20 | } | 20 | } |
| 21 | 21 | ||
| 22 | inline Xbyak::Reg64 IndexToReg64(std::size_t reg_index) { | 22 | constexpr Xbyak::Reg64 IndexToReg64(std::size_t reg_index) { |
| 23 | ASSERT(reg_index < 16); | 23 | ASSERT(reg_index < 16); |
| 24 | return Xbyak::Reg64(static_cast<int>(reg_index)); | 24 | return Xbyak::Reg64(static_cast<int>(reg_index)); |
| 25 | } | 25 | } |
| 26 | 26 | ||
| 27 | inline Xbyak::Xmm IndexToXmm(std::size_t reg_index) { | 27 | constexpr Xbyak::Xmm IndexToXmm(std::size_t reg_index) { |
| 28 | ASSERT(reg_index >= 16 && reg_index < 32); | 28 | ASSERT(reg_index >= 16 && reg_index < 32); |
| 29 | return Xbyak::Xmm(static_cast<int>(reg_index - 16)); | 29 | return Xbyak::Xmm(static_cast<int>(reg_index - 16)); |
| 30 | } | 30 | } |
| 31 | 31 | ||
| 32 | inline Xbyak::Reg IndexToReg(std::size_t reg_index) { | 32 | constexpr Xbyak::Reg IndexToReg(std::size_t reg_index) { |
| 33 | if (reg_index < 16) { | 33 | if (reg_index < 16) { |
| 34 | return IndexToReg64(reg_index); | 34 | return IndexToReg64(reg_index); |
| 35 | } else { | 35 | } else { |
| @@ -45,17 +45,17 @@ inline std::bitset<32> BuildRegSet(std::initializer_list<Xbyak::Reg> regs) { | |||
| 45 | return bits; | 45 | return bits; |
| 46 | } | 46 | } |
| 47 | 47 | ||
| 48 | const std::bitset<32> ABI_ALL_GPRS(0x0000FFFF); | 48 | constexpr inline std::bitset<32> ABI_ALL_GPRS(0x0000FFFF); |
| 49 | const std::bitset<32> ABI_ALL_XMMS(0xFFFF0000); | 49 | constexpr inline std::bitset<32> ABI_ALL_XMMS(0xFFFF0000); |
| 50 | 50 | ||
| 51 | #ifdef _WIN32 | 51 | #ifdef _WIN32 |
| 52 | 52 | ||
| 53 | // Microsoft x64 ABI | 53 | // Microsoft x64 ABI |
| 54 | const Xbyak::Reg ABI_RETURN = Xbyak::util::rax; | 54 | constexpr inline Xbyak::Reg ABI_RETURN = Xbyak::util::rax; |
| 55 | const Xbyak::Reg ABI_PARAM1 = Xbyak::util::rcx; | 55 | constexpr inline Xbyak::Reg ABI_PARAM1 = Xbyak::util::rcx; |
| 56 | const Xbyak::Reg ABI_PARAM2 = Xbyak::util::rdx; | 56 | constexpr inline Xbyak::Reg ABI_PARAM2 = Xbyak::util::rdx; |
| 57 | const Xbyak::Reg ABI_PARAM3 = Xbyak::util::r8; | 57 | constexpr inline Xbyak::Reg ABI_PARAM3 = Xbyak::util::r8; |
| 58 | const Xbyak::Reg ABI_PARAM4 = Xbyak::util::r9; | 58 | constexpr inline Xbyak::Reg ABI_PARAM4 = Xbyak::util::r9; |
| 59 | 59 | ||
| 60 | const std::bitset<32> ABI_ALL_CALLER_SAVED = BuildRegSet({ | 60 | const std::bitset<32> ABI_ALL_CALLER_SAVED = BuildRegSet({ |
| 61 | // GPRs | 61 | // GPRs |
| @@ -102,11 +102,11 @@ constexpr size_t ABI_SHADOW_SPACE = 0x20; | |||
| 102 | #else | 102 | #else |
| 103 | 103 | ||
| 104 | // System V x86-64 ABI | 104 | // System V x86-64 ABI |
| 105 | const Xbyak::Reg ABI_RETURN = Xbyak::util::rax; | 105 | constexpr inline Xbyak::Reg ABI_RETURN = Xbyak::util::rax; |
| 106 | const Xbyak::Reg ABI_PARAM1 = Xbyak::util::rdi; | 106 | constexpr inline Xbyak::Reg ABI_PARAM1 = Xbyak::util::rdi; |
| 107 | const Xbyak::Reg ABI_PARAM2 = Xbyak::util::rsi; | 107 | constexpr inline Xbyak::Reg ABI_PARAM2 = Xbyak::util::rsi; |
| 108 | const Xbyak::Reg ABI_PARAM3 = Xbyak::util::rdx; | 108 | constexpr inline Xbyak::Reg ABI_PARAM3 = Xbyak::util::rdx; |
| 109 | const Xbyak::Reg ABI_PARAM4 = Xbyak::util::rcx; | 109 | constexpr inline Xbyak::Reg ABI_PARAM4 = Xbyak::util::rcx; |
| 110 | 110 | ||
| 111 | const std::bitset<32> ABI_ALL_CALLER_SAVED = BuildRegSet({ | 111 | const std::bitset<32> ABI_ALL_CALLER_SAVED = BuildRegSet({ |
| 112 | // GPRs | 112 | // GPRs |
diff --git a/src/core/CMakeLists.txt b/src/core/CMakeLists.txt index c85c9485f..b96ca9374 100644 --- a/src/core/CMakeLists.txt +++ b/src/core/CMakeLists.txt | |||
| @@ -491,6 +491,7 @@ add_library(core STATIC | |||
| 491 | hle/service/sm/controller.h | 491 | hle/service/sm/controller.h |
| 492 | hle/service/sm/sm.cpp | 492 | hle/service/sm/sm.cpp |
| 493 | hle/service/sm/sm.h | 493 | hle/service/sm/sm.h |
| 494 | hle/service/sockets/blocking_worker.h | ||
| 494 | hle/service/sockets/bsd.cpp | 495 | hle/service/sockets/bsd.cpp |
| 495 | hle/service/sockets/bsd.h | 496 | hle/service/sockets/bsd.h |
| 496 | hle/service/sockets/ethc.cpp | 497 | hle/service/sockets/ethc.cpp |
| @@ -501,6 +502,8 @@ add_library(core STATIC | |||
| 501 | hle/service/sockets/sfdnsres.h | 502 | hle/service/sockets/sfdnsres.h |
| 502 | hle/service/sockets/sockets.cpp | 503 | hle/service/sockets/sockets.cpp |
| 503 | hle/service/sockets/sockets.h | 504 | hle/service/sockets/sockets.h |
| 505 | hle/service/sockets/sockets_translate.cpp | ||
| 506 | hle/service/sockets/sockets_translate.h | ||
| 504 | hle/service/spl/csrng.cpp | 507 | hle/service/spl/csrng.cpp |
| 505 | hle/service/spl/csrng.h | 508 | hle/service/spl/csrng.h |
| 506 | hle/service/spl/module.cpp | 509 | hle/service/spl/module.cpp |
diff --git a/src/core/cpu_manager.cpp b/src/core/cpu_manager.cpp index ef0bae556..688b99eba 100644 --- a/src/core/cpu_manager.cpp +++ b/src/core/cpu_manager.cpp | |||
| @@ -328,7 +328,7 @@ void CpuManager::RunThread(std::size_t core) { | |||
| 328 | system.RegisterCoreThread(core); | 328 | system.RegisterCoreThread(core); |
| 329 | std::string name; | 329 | std::string name; |
| 330 | if (is_multicore) { | 330 | if (is_multicore) { |
| 331 | name = "yuzu:CoreCPUThread_" + std::to_string(core); | 331 | name = "yuzu:CPUCore_" + std::to_string(core); |
| 332 | } else { | 332 | } else { |
| 333 | name = "yuzu:CPUThread"; | 333 | name = "yuzu:CPUThread"; |
| 334 | } | 334 | } |
diff --git a/src/core/crypto/partition_data_manager.cpp b/src/core/crypto/partition_data_manager.cpp index 46136d04a..5f1c86a09 100644 --- a/src/core/crypto/partition_data_manager.cpp +++ b/src/core/crypto/partition_data_manager.cpp | |||
| @@ -26,6 +26,7 @@ | |||
| 26 | #include "core/file_sys/vfs.h" | 26 | #include "core/file_sys/vfs.h" |
| 27 | #include "core/file_sys/vfs_offset.h" | 27 | #include "core/file_sys/vfs_offset.h" |
| 28 | #include "core/file_sys/vfs_vector.h" | 28 | #include "core/file_sys/vfs_vector.h" |
| 29 | #include "core/loader/loader.h" | ||
| 29 | 30 | ||
| 30 | using Common::AsArray; | 31 | using Common::AsArray; |
| 31 | 32 | ||
diff --git a/src/core/file_sys/bis_factory.cpp b/src/core/file_sys/bis_factory.cpp index 9ffda2e14..e04a54c3c 100644 --- a/src/core/file_sys/bis_factory.cpp +++ b/src/core/file_sys/bis_factory.cpp | |||
| @@ -8,7 +8,6 @@ | |||
| 8 | #include "core/file_sys/bis_factory.h" | 8 | #include "core/file_sys/bis_factory.h" |
| 9 | #include "core/file_sys/mode.h" | 9 | #include "core/file_sys/mode.h" |
| 10 | #include "core/file_sys/registered_cache.h" | 10 | #include "core/file_sys/registered_cache.h" |
| 11 | #include "core/settings.h" | ||
| 12 | 11 | ||
| 13 | namespace FileSys { | 12 | namespace FileSys { |
| 14 | 13 | ||
diff --git a/src/core/file_sys/bis_factory.h b/src/core/file_sys/bis_factory.h index 8f0451c98..438d3f8d8 100644 --- a/src/core/file_sys/bis_factory.h +++ b/src/core/file_sys/bis_factory.h | |||
| @@ -6,7 +6,8 @@ | |||
| 6 | 6 | ||
| 7 | #include <memory> | 7 | #include <memory> |
| 8 | 8 | ||
| 9 | #include "core/file_sys/vfs.h" | 9 | #include "common/common_types.h" |
| 10 | #include "core/file_sys/vfs_types.h" | ||
| 10 | 11 | ||
| 11 | namespace FileSys { | 12 | namespace FileSys { |
| 12 | 13 | ||
diff --git a/src/core/file_sys/card_image.cpp b/src/core/file_sys/card_image.cpp index 664a47e7f..956da68f7 100644 --- a/src/core/file_sys/card_image.cpp +++ b/src/core/file_sys/card_image.cpp | |||
| @@ -8,11 +8,11 @@ | |||
| 8 | #include <fmt/ostream.h> | 8 | #include <fmt/ostream.h> |
| 9 | 9 | ||
| 10 | #include "common/logging/log.h" | 10 | #include "common/logging/log.h" |
| 11 | #include "core/crypto/key_manager.h" | ||
| 11 | #include "core/file_sys/card_image.h" | 12 | #include "core/file_sys/card_image.h" |
| 12 | #include "core/file_sys/content_archive.h" | 13 | #include "core/file_sys/content_archive.h" |
| 13 | #include "core/file_sys/nca_metadata.h" | 14 | #include "core/file_sys/nca_metadata.h" |
| 14 | #include "core/file_sys/partition_filesystem.h" | 15 | #include "core/file_sys/partition_filesystem.h" |
| 15 | #include "core/file_sys/romfs.h" | ||
| 16 | #include "core/file_sys/submission_package.h" | 16 | #include "core/file_sys/submission_package.h" |
| 17 | #include "core/file_sys/vfs_concat.h" | 17 | #include "core/file_sys/vfs_concat.h" |
| 18 | #include "core/file_sys/vfs_offset.h" | 18 | #include "core/file_sys/vfs_offset.h" |
| @@ -31,7 +31,8 @@ constexpr std::array partition_names{ | |||
| 31 | 31 | ||
| 32 | XCI::XCI(VirtualFile file_) | 32 | XCI::XCI(VirtualFile file_) |
| 33 | : file(std::move(file_)), program_nca_status{Loader::ResultStatus::ErrorXCIMissingProgramNCA}, | 33 | : file(std::move(file_)), program_nca_status{Loader::ResultStatus::ErrorXCIMissingProgramNCA}, |
| 34 | partitions(partition_names.size()), partitions_raw(partition_names.size()) { | 34 | partitions(partition_names.size()), |
| 35 | partitions_raw(partition_names.size()), keys{Core::Crypto::KeyManager::Instance()} { | ||
| 35 | if (file->ReadObject(&header) != sizeof(GamecardHeader)) { | 36 | if (file->ReadObject(&header) != sizeof(GamecardHeader)) { |
| 36 | status = Loader::ResultStatus::ErrorBadXCIHeader; | 37 | status = Loader::ResultStatus::ErrorBadXCIHeader; |
| 37 | return; | 38 | return; |
diff --git a/src/core/file_sys/card_image.h b/src/core/file_sys/card_image.h index e1b136426..2d0a0f285 100644 --- a/src/core/file_sys/card_image.h +++ b/src/core/file_sys/card_image.h | |||
| @@ -9,9 +9,12 @@ | |||
| 9 | #include <vector> | 9 | #include <vector> |
| 10 | #include "common/common_types.h" | 10 | #include "common/common_types.h" |
| 11 | #include "common/swap.h" | 11 | #include "common/swap.h" |
| 12 | #include "core/crypto/key_manager.h" | ||
| 13 | #include "core/file_sys/vfs.h" | 12 | #include "core/file_sys/vfs.h" |
| 14 | 13 | ||
| 14 | namespace Core::Crypto { | ||
| 15 | class KeyManager; | ||
| 16 | } | ||
| 17 | |||
| 15 | namespace Loader { | 18 | namespace Loader { |
| 16 | enum class ResultStatus : u16; | 19 | enum class ResultStatus : u16; |
| 17 | } | 20 | } |
| @@ -140,6 +143,6 @@ private: | |||
| 140 | 143 | ||
| 141 | u64 update_normal_partition_end; | 144 | u64 update_normal_partition_end; |
| 142 | 145 | ||
| 143 | Core::Crypto::KeyManager& keys = Core::Crypto::KeyManager::Instance(); | 146 | Core::Crypto::KeyManager& keys; |
| 144 | }; | 147 | }; |
| 145 | } // namespace FileSys | 148 | } // namespace FileSys |
diff --git a/src/core/file_sys/content_archive.cpp b/src/core/file_sys/content_archive.cpp index 5039341c7..426fb6bb5 100644 --- a/src/core/file_sys/content_archive.cpp +++ b/src/core/file_sys/content_archive.cpp | |||
| @@ -10,10 +10,10 @@ | |||
| 10 | #include "common/logging/log.h" | 10 | #include "common/logging/log.h" |
| 11 | #include "core/crypto/aes_util.h" | 11 | #include "core/crypto/aes_util.h" |
| 12 | #include "core/crypto/ctr_encryption_layer.h" | 12 | #include "core/crypto/ctr_encryption_layer.h" |
| 13 | #include "core/crypto/key_manager.h" | ||
| 13 | #include "core/file_sys/content_archive.h" | 14 | #include "core/file_sys/content_archive.h" |
| 14 | #include "core/file_sys/nca_patch.h" | 15 | #include "core/file_sys/nca_patch.h" |
| 15 | #include "core/file_sys/partition_filesystem.h" | 16 | #include "core/file_sys/partition_filesystem.h" |
| 16 | #include "core/file_sys/romfs.h" | ||
| 17 | #include "core/file_sys/vfs_offset.h" | 17 | #include "core/file_sys/vfs_offset.h" |
| 18 | #include "core/loader/loader.h" | 18 | #include "core/loader/loader.h" |
| 19 | 19 | ||
| @@ -119,7 +119,8 @@ static bool IsValidNCA(const NCAHeader& header) { | |||
| 119 | } | 119 | } |
| 120 | 120 | ||
| 121 | NCA::NCA(VirtualFile file_, VirtualFile bktr_base_romfs_, u64 bktr_base_ivfc_offset) | 121 | NCA::NCA(VirtualFile file_, VirtualFile bktr_base_romfs_, u64 bktr_base_ivfc_offset) |
| 122 | : file(std::move(file_)), bktr_base_romfs(std::move(bktr_base_romfs_)) { | 122 | : file(std::move(file_)), |
| 123 | bktr_base_romfs(std::move(bktr_base_romfs_)), keys{Core::Crypto::KeyManager::Instance()} { | ||
| 123 | if (file == nullptr) { | 124 | if (file == nullptr) { |
| 124 | status = Loader::ResultStatus::ErrorNullFile; | 125 | status = Loader::ResultStatus::ErrorNullFile; |
| 125 | return; | 126 | return; |
diff --git a/src/core/file_sys/content_archive.h b/src/core/file_sys/content_archive.h index d25cbcf91..69292232a 100644 --- a/src/core/file_sys/content_archive.h +++ b/src/core/file_sys/content_archive.h | |||
| @@ -158,7 +158,7 @@ private: | |||
| 158 | bool encrypted = false; | 158 | bool encrypted = false; |
| 159 | bool is_update = false; | 159 | bool is_update = false; |
| 160 | 160 | ||
| 161 | Core::Crypto::KeyManager& keys = Core::Crypto::KeyManager::Instance(); | 161 | Core::Crypto::KeyManager& keys; |
| 162 | }; | 162 | }; |
| 163 | 163 | ||
| 164 | } // namespace FileSys | 164 | } // namespace FileSys |
diff --git a/src/core/file_sys/control_metadata.cpp b/src/core/file_sys/control_metadata.cpp index 63cd2eead..b0a130345 100644 --- a/src/core/file_sys/control_metadata.cpp +++ b/src/core/file_sys/control_metadata.cpp | |||
| @@ -5,6 +5,7 @@ | |||
| 5 | #include "common/string_util.h" | 5 | #include "common/string_util.h" |
| 6 | #include "common/swap.h" | 6 | #include "common/swap.h" |
| 7 | #include "core/file_sys/control_metadata.h" | 7 | #include "core/file_sys/control_metadata.h" |
| 8 | #include "core/file_sys/vfs.h" | ||
| 8 | 9 | ||
| 9 | namespace FileSys { | 10 | namespace FileSys { |
| 10 | 11 | ||
diff --git a/src/core/file_sys/control_metadata.h b/src/core/file_sys/control_metadata.h index e37b2fadf..9ab86e35b 100644 --- a/src/core/file_sys/control_metadata.h +++ b/src/core/file_sys/control_metadata.h | |||
| @@ -10,7 +10,7 @@ | |||
| 10 | #include "common/common_funcs.h" | 10 | #include "common/common_funcs.h" |
| 11 | #include "common/common_types.h" | 11 | #include "common/common_types.h" |
| 12 | #include "common/swap.h" | 12 | #include "common/swap.h" |
| 13 | #include "core/file_sys/vfs.h" | 13 | #include "core/file_sys/vfs_types.h" |
| 14 | 14 | ||
| 15 | namespace FileSys { | 15 | namespace FileSys { |
| 16 | 16 | ||
diff --git a/src/core/file_sys/kernel_executable.cpp b/src/core/file_sys/kernel_executable.cpp index 76313679d..ef93ef3ed 100644 --- a/src/core/file_sys/kernel_executable.cpp +++ b/src/core/file_sys/kernel_executable.cpp | |||
| @@ -2,9 +2,12 @@ | |||
| 2 | // Licensed under GPLv2 or any later version | 2 | // Licensed under GPLv2 or any later version |
| 3 | // Refer to the license.txt file included. | 3 | // Refer to the license.txt file included. |
| 4 | 4 | ||
| 5 | #include <cstring> | ||
| 6 | |||
| 5 | #include "common/string_util.h" | 7 | #include "common/string_util.h" |
| 6 | #include "core/file_sys/kernel_executable.h" | 8 | #include "core/file_sys/kernel_executable.h" |
| 7 | #include "core/file_sys/vfs_offset.h" | 9 | #include "core/file_sys/vfs_offset.h" |
| 10 | #include "core/loader/loader.h" | ||
| 8 | 11 | ||
| 9 | namespace FileSys { | 12 | namespace FileSys { |
| 10 | 13 | ||
diff --git a/src/core/file_sys/kernel_executable.h b/src/core/file_sys/kernel_executable.h index 324a57384..044c554d3 100644 --- a/src/core/file_sys/kernel_executable.h +++ b/src/core/file_sys/kernel_executable.h | |||
| @@ -4,10 +4,17 @@ | |||
| 4 | 4 | ||
| 5 | #pragma once | 5 | #pragma once |
| 6 | 6 | ||
| 7 | #include <array> | ||
| 8 | #include <vector> | ||
| 9 | |||
| 7 | #include "common/common_funcs.h" | 10 | #include "common/common_funcs.h" |
| 11 | #include "common/common_types.h" | ||
| 8 | #include "common/swap.h" | 12 | #include "common/swap.h" |
| 9 | #include "core/file_sys/vfs_types.h" | 13 | #include "core/file_sys/vfs_types.h" |
| 10 | #include "core/loader/loader.h" | 14 | |
| 15 | namespace Loader { | ||
| 16 | enum class ResultStatus : u16; | ||
| 17 | } | ||
| 11 | 18 | ||
| 12 | namespace FileSys { | 19 | namespace FileSys { |
| 13 | 20 | ||
diff --git a/src/core/file_sys/nca_metadata.cpp b/src/core/file_sys/nca_metadata.cpp index 93d0df6b9..2d1476e3a 100644 --- a/src/core/file_sys/nca_metadata.cpp +++ b/src/core/file_sys/nca_metadata.cpp | |||
| @@ -7,6 +7,7 @@ | |||
| 7 | #include "common/logging/log.h" | 7 | #include "common/logging/log.h" |
| 8 | #include "common/swap.h" | 8 | #include "common/swap.h" |
| 9 | #include "core/file_sys/nca_metadata.h" | 9 | #include "core/file_sys/nca_metadata.h" |
| 10 | #include "core/file_sys/vfs.h" | ||
| 10 | 11 | ||
| 11 | namespace FileSys { | 12 | namespace FileSys { |
| 12 | 13 | ||
diff --git a/src/core/file_sys/nca_metadata.h b/src/core/file_sys/nca_metadata.h index 1f82fff0a..53535e5f5 100644 --- a/src/core/file_sys/nca_metadata.h +++ b/src/core/file_sys/nca_metadata.h | |||
| @@ -10,7 +10,7 @@ | |||
| 10 | #include "common/common_funcs.h" | 10 | #include "common/common_funcs.h" |
| 11 | #include "common/common_types.h" | 11 | #include "common/common_types.h" |
| 12 | #include "common/swap.h" | 12 | #include "common/swap.h" |
| 13 | #include "core/file_sys/vfs.h" | 13 | #include "core/file_sys/vfs_types.h" |
| 14 | 14 | ||
| 15 | namespace FileSys { | 15 | namespace FileSys { |
| 16 | class CNMT; | 16 | class CNMT; |
diff --git a/src/core/file_sys/partition_filesystem.cpp b/src/core/file_sys/partition_filesystem.cpp index 846986736..48a2ed4d4 100644 --- a/src/core/file_sys/partition_filesystem.cpp +++ b/src/core/file_sys/partition_filesystem.cpp | |||
| @@ -21,7 +21,7 @@ bool PartitionFilesystem::Header::HasValidMagicValue() const { | |||
| 21 | magic == Common::MakeMagic('P', 'F', 'S', '0'); | 21 | magic == Common::MakeMagic('P', 'F', 'S', '0'); |
| 22 | } | 22 | } |
| 23 | 23 | ||
| 24 | PartitionFilesystem::PartitionFilesystem(std::shared_ptr<VfsFile> file) { | 24 | PartitionFilesystem::PartitionFilesystem(VirtualFile file) { |
| 25 | // At least be as large as the header | 25 | // At least be as large as the header |
| 26 | if (file->GetSize() < sizeof(Header)) { | 26 | if (file->GetSize() < sizeof(Header)) { |
| 27 | status = Loader::ResultStatus::ErrorBadPFSHeader; | 27 | status = Loader::ResultStatus::ErrorBadPFSHeader; |
| @@ -89,11 +89,11 @@ std::map<std::string, u64> PartitionFilesystem::GetFileSizes() const { | |||
| 89 | return sizes; | 89 | return sizes; |
| 90 | } | 90 | } |
| 91 | 91 | ||
| 92 | std::vector<std::shared_ptr<VfsFile>> PartitionFilesystem::GetFiles() const { | 92 | std::vector<VirtualFile> PartitionFilesystem::GetFiles() const { |
| 93 | return pfs_files; | 93 | return pfs_files; |
| 94 | } | 94 | } |
| 95 | 95 | ||
| 96 | std::vector<std::shared_ptr<VfsDirectory>> PartitionFilesystem::GetSubdirectories() const { | 96 | std::vector<VirtualDir> PartitionFilesystem::GetSubdirectories() const { |
| 97 | return {}; | 97 | return {}; |
| 98 | } | 98 | } |
| 99 | 99 | ||
| @@ -101,7 +101,7 @@ std::string PartitionFilesystem::GetName() const { | |||
| 101 | return is_hfs ? "HFS0" : "PFS0"; | 101 | return is_hfs ? "HFS0" : "PFS0"; |
| 102 | } | 102 | } |
| 103 | 103 | ||
| 104 | std::shared_ptr<VfsDirectory> PartitionFilesystem::GetParentDirectory() const { | 104 | VirtualDir PartitionFilesystem::GetParentDirectory() const { |
| 105 | // TODO(DarkLordZach): Add support for nested containers. | 105 | // TODO(DarkLordZach): Add support for nested containers. |
| 106 | return nullptr; | 106 | return nullptr; |
| 107 | } | 107 | } |
diff --git a/src/core/file_sys/partition_filesystem.h b/src/core/file_sys/partition_filesystem.h index 279193b19..0f831148e 100644 --- a/src/core/file_sys/partition_filesystem.h +++ b/src/core/file_sys/partition_filesystem.h | |||
| @@ -24,7 +24,7 @@ namespace FileSys { | |||
| 24 | */ | 24 | */ |
| 25 | class PartitionFilesystem : public ReadOnlyVfsDirectory { | 25 | class PartitionFilesystem : public ReadOnlyVfsDirectory { |
| 26 | public: | 26 | public: |
| 27 | explicit PartitionFilesystem(std::shared_ptr<VfsFile> file); | 27 | explicit PartitionFilesystem(VirtualFile file); |
| 28 | ~PartitionFilesystem() override; | 28 | ~PartitionFilesystem() override; |
| 29 | 29 | ||
| 30 | Loader::ResultStatus GetStatus() const; | 30 | Loader::ResultStatus GetStatus() const; |
| @@ -32,10 +32,10 @@ public: | |||
| 32 | std::map<std::string, u64> GetFileOffsets() const; | 32 | std::map<std::string, u64> GetFileOffsets() const; |
| 33 | std::map<std::string, u64> GetFileSizes() const; | 33 | std::map<std::string, u64> GetFileSizes() const; |
| 34 | 34 | ||
| 35 | std::vector<std::shared_ptr<VfsFile>> GetFiles() const override; | 35 | std::vector<VirtualFile> GetFiles() const override; |
| 36 | std::vector<std::shared_ptr<VfsDirectory>> GetSubdirectories() const override; | 36 | std::vector<VirtualDir> GetSubdirectories() const override; |
| 37 | std::string GetName() const override; | 37 | std::string GetName() const override; |
| 38 | std::shared_ptr<VfsDirectory> GetParentDirectory() const override; | 38 | VirtualDir GetParentDirectory() const override; |
| 39 | void PrintDebugInfo() const; | 39 | void PrintDebugInfo() const; |
| 40 | 40 | ||
| 41 | private: | 41 | private: |
diff --git a/src/core/file_sys/patch_manager.cpp b/src/core/file_sys/patch_manager.cpp index 729dbb5f4..c228d253e 100644 --- a/src/core/file_sys/patch_manager.cpp +++ b/src/core/file_sys/patch_manager.cpp | |||
| @@ -49,8 +49,7 @@ std::string FormatTitleVersion(u32 version, TitleVersionFormat format) { | |||
| 49 | return fmt::format("v{}.{}.{}", bytes[3], bytes[2], bytes[1]); | 49 | return fmt::format("v{}.{}.{}", bytes[3], bytes[2], bytes[1]); |
| 50 | } | 50 | } |
| 51 | 51 | ||
| 52 | std::shared_ptr<VfsDirectory> FindSubdirectoryCaseless(const std::shared_ptr<VfsDirectory> dir, | 52 | VirtualDir FindSubdirectoryCaseless(const VirtualDir dir, std::string_view name) { |
| 53 | std::string_view name) { | ||
| 54 | #ifdef _WIN32 | 53 | #ifdef _WIN32 |
| 55 | return dir->GetSubdirectory(name); | 54 | return dir->GetSubdirectory(name); |
| 56 | #else | 55 | #else |
diff --git a/src/core/file_sys/patch_manager.h b/src/core/file_sys/patch_manager.h index f4cb918dd..532f4995f 100644 --- a/src/core/file_sys/patch_manager.h +++ b/src/core/file_sys/patch_manager.h | |||
| @@ -6,10 +6,11 @@ | |||
| 6 | 6 | ||
| 7 | #include <map> | 7 | #include <map> |
| 8 | #include <memory> | 8 | #include <memory> |
| 9 | #include <optional> | ||
| 9 | #include <string> | 10 | #include <string> |
| 10 | #include "common/common_types.h" | 11 | #include "common/common_types.h" |
| 11 | #include "core/file_sys/nca_metadata.h" | 12 | #include "core/file_sys/nca_metadata.h" |
| 12 | #include "core/file_sys/vfs.h" | 13 | #include "core/file_sys/vfs_types.h" |
| 13 | #include "core/memory/dmnt_cheat_types.h" | 14 | #include "core/memory/dmnt_cheat_types.h" |
| 14 | 15 | ||
| 15 | namespace Core { | 16 | namespace Core { |
| @@ -31,8 +32,7 @@ std::string FormatTitleVersion(u32 version, | |||
| 31 | 32 | ||
| 32 | // Returns a directory with name matching name case-insensitive. Returns nullptr if directory | 33 | // Returns a directory with name matching name case-insensitive. Returns nullptr if directory |
| 33 | // doesn't have a directory with name. | 34 | // doesn't have a directory with name. |
| 34 | std::shared_ptr<VfsDirectory> FindSubdirectoryCaseless(const std::shared_ptr<VfsDirectory> dir, | 35 | VirtualDir FindSubdirectoryCaseless(VirtualDir dir, std::string_view name); |
| 35 | std::string_view name); | ||
| 36 | 36 | ||
| 37 | // A centralized class to manage patches to games. | 37 | // A centralized class to manage patches to games. |
| 38 | class PatchManager { | 38 | class PatchManager { |
diff --git a/src/core/file_sys/program_metadata.cpp b/src/core/file_sys/program_metadata.cpp index 43169bf9f..9cf49bf44 100644 --- a/src/core/file_sys/program_metadata.cpp +++ b/src/core/file_sys/program_metadata.cpp | |||
| @@ -7,6 +7,7 @@ | |||
| 7 | 7 | ||
| 8 | #include "common/logging/log.h" | 8 | #include "common/logging/log.h" |
| 9 | #include "core/file_sys/program_metadata.h" | 9 | #include "core/file_sys/program_metadata.h" |
| 10 | #include "core/file_sys/vfs.h" | ||
| 10 | #include "core/loader/loader.h" | 11 | #include "core/loader/loader.h" |
| 11 | 12 | ||
| 12 | namespace FileSys { | 13 | namespace FileSys { |
diff --git a/src/core/file_sys/program_metadata.h b/src/core/file_sys/program_metadata.h index 35069972b..455532567 100644 --- a/src/core/file_sys/program_metadata.h +++ b/src/core/file_sys/program_metadata.h | |||
| @@ -9,7 +9,7 @@ | |||
| 9 | #include "common/bit_field.h" | 9 | #include "common/bit_field.h" |
| 10 | #include "common/common_types.h" | 10 | #include "common/common_types.h" |
| 11 | #include "common/swap.h" | 11 | #include "common/swap.h" |
| 12 | #include "core/file_sys/vfs.h" | 12 | #include "core/file_sys/vfs_types.h" |
| 13 | 13 | ||
| 14 | namespace Loader { | 14 | namespace Loader { |
| 15 | enum class ResultStatus : u16; | 15 | enum class ResultStatus : u16; |
diff --git a/src/core/file_sys/romfs.h b/src/core/file_sys/romfs.h index 2fd07ed04..82e683782 100644 --- a/src/core/file_sys/romfs.h +++ b/src/core/file_sys/romfs.h | |||
| @@ -4,7 +4,6 @@ | |||
| 4 | 4 | ||
| 5 | #pragma once | 5 | #pragma once |
| 6 | 6 | ||
| 7 | #include <array> | ||
| 8 | #include "core/file_sys/vfs.h" | 7 | #include "core/file_sys/vfs.h" |
| 9 | 8 | ||
| 10 | namespace FileSys { | 9 | namespace FileSys { |
diff --git a/src/core/file_sys/sdmc_factory.cpp b/src/core/file_sys/sdmc_factory.cpp index 6f732e4d8..cb56d8f2d 100644 --- a/src/core/file_sys/sdmc_factory.cpp +++ b/src/core/file_sys/sdmc_factory.cpp | |||
| @@ -5,8 +5,8 @@ | |||
| 5 | #include <memory> | 5 | #include <memory> |
| 6 | #include "core/file_sys/registered_cache.h" | 6 | #include "core/file_sys/registered_cache.h" |
| 7 | #include "core/file_sys/sdmc_factory.h" | 7 | #include "core/file_sys/sdmc_factory.h" |
| 8 | #include "core/file_sys/vfs.h" | ||
| 8 | #include "core/file_sys/xts_archive.h" | 9 | #include "core/file_sys/xts_archive.h" |
| 9 | #include "core/settings.h" | ||
| 10 | 10 | ||
| 11 | namespace FileSys { | 11 | namespace FileSys { |
| 12 | 12 | ||
diff --git a/src/core/file_sys/sdmc_factory.h b/src/core/file_sys/sdmc_factory.h index 42dc4e08a..2bb92ba93 100644 --- a/src/core/file_sys/sdmc_factory.h +++ b/src/core/file_sys/sdmc_factory.h | |||
| @@ -5,7 +5,7 @@ | |||
| 5 | #pragma once | 5 | #pragma once |
| 6 | 6 | ||
| 7 | #include <memory> | 7 | #include <memory> |
| 8 | #include "core/file_sys/vfs.h" | 8 | #include "core/file_sys/vfs_types.h" |
| 9 | #include "core/hle/result.h" | 9 | #include "core/hle/result.h" |
| 10 | 10 | ||
| 11 | namespace FileSys { | 11 | namespace FileSys { |
diff --git a/src/core/file_sys/submission_package.cpp b/src/core/file_sys/submission_package.cpp index 175a8266a..b9ce93b7c 100644 --- a/src/core/file_sys/submission_package.cpp +++ b/src/core/file_sys/submission_package.cpp | |||
| @@ -54,7 +54,7 @@ void SetTicketKeys(const std::vector<VirtualFile>& files) { | |||
| 54 | 54 | ||
| 55 | NSP::NSP(VirtualFile file_) | 55 | NSP::NSP(VirtualFile file_) |
| 56 | : file(std::move(file_)), status{Loader::ResultStatus::Success}, | 56 | : file(std::move(file_)), status{Loader::ResultStatus::Success}, |
| 57 | pfs(std::make_shared<PartitionFilesystem>(file)) { | 57 | pfs(std::make_shared<PartitionFilesystem>(file)), keys{Core::Crypto::KeyManager::Instance()} { |
| 58 | if (pfs->GetStatus() != Loader::ResultStatus::Success) { | 58 | if (pfs->GetStatus() != Loader::ResultStatus::Success) { |
| 59 | status = pfs->GetStatus(); | 59 | status = pfs->GetStatus(); |
| 60 | return; | 60 | return; |
diff --git a/src/core/file_sys/submission_package.h b/src/core/file_sys/submission_package.h index cf89de6a9..2db5e46b8 100644 --- a/src/core/file_sys/submission_package.h +++ b/src/core/file_sys/submission_package.h | |||
| @@ -10,6 +10,10 @@ | |||
| 10 | #include "common/common_types.h" | 10 | #include "common/common_types.h" |
| 11 | #include "core/file_sys/vfs.h" | 11 | #include "core/file_sys/vfs.h" |
| 12 | 12 | ||
| 13 | namespace Core::Crypto { | ||
| 14 | class KeyManager; | ||
| 15 | } | ||
| 16 | |||
| 13 | namespace Loader { | 17 | namespace Loader { |
| 14 | enum class ResultStatus : u16; | 18 | enum class ResultStatus : u16; |
| 15 | } | 19 | } |
| @@ -73,7 +77,7 @@ private: | |||
| 73 | std::map<u64, std::map<std::pair<TitleType, ContentRecordType>, std::shared_ptr<NCA>>> ncas; | 77 | std::map<u64, std::map<std::pair<TitleType, ContentRecordType>, std::shared_ptr<NCA>>> ncas; |
| 74 | std::vector<VirtualFile> ticket_files; | 78 | std::vector<VirtualFile> ticket_files; |
| 75 | 79 | ||
| 76 | Core::Crypto::KeyManager& keys = Core::Crypto::KeyManager::Instance(); | 80 | Core::Crypto::KeyManager& keys; |
| 77 | 81 | ||
| 78 | VirtualFile romfs; | 82 | VirtualFile romfs; |
| 79 | VirtualDir exefs; | 83 | VirtualDir exefs; |
diff --git a/src/core/file_sys/xts_archive.cpp b/src/core/file_sys/xts_archive.cpp index ccf5966d0..24c58e7ae 100644 --- a/src/core/file_sys/xts_archive.cpp +++ b/src/core/file_sys/xts_archive.cpp | |||
| @@ -15,8 +15,9 @@ | |||
| 15 | #include "common/hex_util.h" | 15 | #include "common/hex_util.h" |
| 16 | #include "common/string_util.h" | 16 | #include "common/string_util.h" |
| 17 | #include "core/crypto/aes_util.h" | 17 | #include "core/crypto/aes_util.h" |
| 18 | #include "core/crypto/key_manager.h" | ||
| 18 | #include "core/crypto/xts_encryption_layer.h" | 19 | #include "core/crypto/xts_encryption_layer.h" |
| 19 | #include "core/file_sys/partition_filesystem.h" | 20 | #include "core/file_sys/content_archive.h" |
| 20 | #include "core/file_sys/vfs_offset.h" | 21 | #include "core/file_sys/vfs_offset.h" |
| 21 | #include "core/file_sys/xts_archive.h" | 22 | #include "core/file_sys/xts_archive.h" |
| 22 | #include "core/loader/loader.h" | 23 | #include "core/loader/loader.h" |
| @@ -43,7 +44,9 @@ static bool CalculateHMAC256(Destination* out, const SourceKey* key, std::size_t | |||
| 43 | return true; | 44 | return true; |
| 44 | } | 45 | } |
| 45 | 46 | ||
| 46 | NAX::NAX(VirtualFile file_) : header(std::make_unique<NAXHeader>()), file(std::move(file_)) { | 47 | NAX::NAX(VirtualFile file_) |
| 48 | : header(std::make_unique<NAXHeader>()), | ||
| 49 | file(std::move(file_)), keys{Core::Crypto::KeyManager::Instance()} { | ||
| 47 | std::string path = Common::FS::SanitizePath(file->GetFullPath()); | 50 | std::string path = Common::FS::SanitizePath(file->GetFullPath()); |
| 48 | static const std::regex nax_path_regex("/registered/(000000[0-9A-F]{2})/([0-9A-F]{32})\\.nca", | 51 | static const std::regex nax_path_regex("/registered/(000000[0-9A-F]{2})/([0-9A-F]{32})\\.nca", |
| 49 | std::regex_constants::ECMAScript | | 52 | std::regex_constants::ECMAScript | |
| @@ -60,7 +63,8 @@ NAX::NAX(VirtualFile file_) : header(std::make_unique<NAXHeader>()), file(std::m | |||
| 60 | } | 63 | } |
| 61 | 64 | ||
| 62 | NAX::NAX(VirtualFile file_, std::array<u8, 0x10> nca_id) | 65 | NAX::NAX(VirtualFile file_, std::array<u8, 0x10> nca_id) |
| 63 | : header(std::make_unique<NAXHeader>()), file(std::move(file_)) { | 66 | : header(std::make_unique<NAXHeader>()), |
| 67 | file(std::move(file_)), keys{Core::Crypto::KeyManager::Instance()} { | ||
| 64 | Core::Crypto::SHA256Hash hash{}; | 68 | Core::Crypto::SHA256Hash hash{}; |
| 65 | mbedtls_sha256_ret(nca_id.data(), nca_id.size(), hash.data(), 0); | 69 | mbedtls_sha256_ret(nca_id.data(), nca_id.size(), hash.data(), 0); |
| 66 | status = Parse(fmt::format("/registered/000000{:02X}/{}.nca", hash[0], | 70 | status = Parse(fmt::format("/registered/000000{:02X}/{}.nca", hash[0], |
diff --git a/src/core/file_sys/xts_archive.h b/src/core/file_sys/xts_archive.h index 563531bb6..c472e226e 100644 --- a/src/core/file_sys/xts_archive.h +++ b/src/core/file_sys/xts_archive.h | |||
| @@ -9,12 +9,16 @@ | |||
| 9 | #include "common/common_types.h" | 9 | #include "common/common_types.h" |
| 10 | #include "common/swap.h" | 10 | #include "common/swap.h" |
| 11 | #include "core/crypto/key_manager.h" | 11 | #include "core/crypto/key_manager.h" |
| 12 | #include "core/file_sys/content_archive.h" | ||
| 13 | #include "core/file_sys/vfs.h" | 12 | #include "core/file_sys/vfs.h" |
| 14 | #include "core/loader/loader.h" | 13 | |
| 14 | namespace Loader { | ||
| 15 | enum class ResultStatus : u16; | ||
| 16 | } | ||
| 15 | 17 | ||
| 16 | namespace FileSys { | 18 | namespace FileSys { |
| 17 | 19 | ||
| 20 | class NCA; | ||
| 21 | |||
| 18 | struct NAXHeader { | 22 | struct NAXHeader { |
| 19 | std::array<u8, 0x20> hmac; | 23 | std::array<u8, 0x20> hmac; |
| 20 | u64_le magic; | 24 | u64_le magic; |
| @@ -62,6 +66,6 @@ private: | |||
| 62 | 66 | ||
| 63 | VirtualFile dec_file; | 67 | VirtualFile dec_file; |
| 64 | 68 | ||
| 65 | Core::Crypto::KeyManager& keys = Core::Crypto::KeyManager::Instance(); | 69 | Core::Crypto::KeyManager& keys; |
| 66 | }; | 70 | }; |
| 67 | } // namespace FileSys | 71 | } // namespace FileSys |
diff --git a/src/core/frontend/framebuffer_layout.h b/src/core/frontend/framebuffer_layout.h index 91ecc30ab..e2e3bbbb3 100644 --- a/src/core/frontend/framebuffer_layout.h +++ b/src/core/frontend/framebuffer_layout.h | |||
| @@ -4,6 +4,7 @@ | |||
| 4 | 4 | ||
| 5 | #pragma once | 5 | #pragma once |
| 6 | 6 | ||
| 7 | #include "common/common_types.h" | ||
| 7 | #include "common/math_util.h" | 8 | #include "common/math_util.h" |
| 8 | 9 | ||
| 9 | namespace Layout { | 10 | namespace Layout { |
diff --git a/src/core/hle/kernel/kernel.cpp b/src/core/hle/kernel/kernel.cpp index cabe8d418..f2b0fe2fd 100644 --- a/src/core/hle/kernel/kernel.cpp +++ b/src/core/hle/kernel/kernel.cpp | |||
| @@ -219,6 +219,7 @@ struct KernelCore::Impl { | |||
| 219 | return static_cast<u32>(system.GetCpuManager().CurrentCore()); | 219 | return static_cast<u32>(system.GetCpuManager().CurrentCore()); |
| 220 | } | 220 | } |
| 221 | } | 221 | } |
| 222 | std::unique_lock lock{register_thread_mutex}; | ||
| 222 | const auto it = host_thread_ids.find(this_id); | 223 | const auto it = host_thread_ids.find(this_id); |
| 223 | if (it == host_thread_ids.end()) { | 224 | if (it == host_thread_ids.end()) { |
| 224 | return Core::INVALID_HOST_THREAD_ID; | 225 | return Core::INVALID_HOST_THREAD_ID; |
| @@ -324,7 +325,7 @@ struct KernelCore::Impl { | |||
| 324 | std::unordered_map<std::thread::id, u32> host_thread_ids; | 325 | std::unordered_map<std::thread::id, u32> host_thread_ids; |
| 325 | u32 registered_thread_ids{Core::Hardware::NUM_CPU_CORES}; | 326 | u32 registered_thread_ids{Core::Hardware::NUM_CPU_CORES}; |
| 326 | std::bitset<Core::Hardware::NUM_CPU_CORES> registered_core_threads; | 327 | std::bitset<Core::Hardware::NUM_CPU_CORES> registered_core_threads; |
| 327 | std::mutex register_thread_mutex; | 328 | mutable std::mutex register_thread_mutex; |
| 328 | 329 | ||
| 329 | // Kernel memory management | 330 | // Kernel memory management |
| 330 | std::unique_ptr<Memory::MemoryManager> memory_manager; | 331 | std::unique_ptr<Memory::MemoryManager> memory_manager; |
diff --git a/src/core/hle/kernel/scheduler.cpp b/src/core/hle/kernel/scheduler.cpp index a4b234424..5cbd3b912 100644 --- a/src/core/hle/kernel/scheduler.cpp +++ b/src/core/hle/kernel/scheduler.cpp | |||
| @@ -756,7 +756,11 @@ void Scheduler::SwitchToCurrent() { | |||
| 756 | current_thread = selected_thread; | 756 | current_thread = selected_thread; |
| 757 | is_context_switch_pending = false; | 757 | is_context_switch_pending = false; |
| 758 | } | 758 | } |
| 759 | while (!is_context_switch_pending) { | 759 | const auto is_switch_pending = [this] { |
| 760 | std::scoped_lock lock{guard}; | ||
| 761 | return is_context_switch_pending; | ||
| 762 | }; | ||
| 763 | do { | ||
| 760 | if (current_thread != nullptr && !current_thread->IsHLEThread()) { | 764 | if (current_thread != nullptr && !current_thread->IsHLEThread()) { |
| 761 | current_thread->context_guard.lock(); | 765 | current_thread->context_guard.lock(); |
| 762 | if (!current_thread->IsRunnable()) { | 766 | if (!current_thread->IsRunnable()) { |
| @@ -775,7 +779,7 @@ void Scheduler::SwitchToCurrent() { | |||
| 775 | next_context = &idle_thread->GetHostContext(); | 779 | next_context = &idle_thread->GetHostContext(); |
| 776 | } | 780 | } |
| 777 | Common::Fiber::YieldTo(switch_fiber, *next_context); | 781 | Common::Fiber::YieldTo(switch_fiber, *next_context); |
| 778 | } | 782 | } while (!is_switch_pending()); |
| 779 | } | 783 | } |
| 780 | } | 784 | } |
| 781 | 785 | ||
diff --git a/src/core/hle/service/filesystem/fsp_srv.cpp b/src/core/hle/service/filesystem/fsp_srv.cpp index 26fd87f58..649128be4 100644 --- a/src/core/hle/service/filesystem/fsp_srv.cpp +++ b/src/core/hle/service/filesystem/fsp_srv.cpp | |||
| @@ -844,8 +844,7 @@ void FSP_SRV::OpenSaveDataFileSystem(Kernel::HLERequestContext& ctx) { | |||
| 844 | return; | 844 | return; |
| 845 | } | 845 | } |
| 846 | 846 | ||
| 847 | FileSys::StorageId id; | 847 | FileSys::StorageId id{}; |
| 848 | |||
| 849 | switch (parameters.space_id) { | 848 | switch (parameters.space_id) { |
| 850 | case FileSys::SaveDataSpaceId::NandUser: | 849 | case FileSys::SaveDataSpaceId::NandUser: |
| 851 | id = FileSys::StorageId::NandUser; | 850 | id = FileSys::StorageId::NandUser; |
| @@ -857,6 +856,10 @@ void FSP_SRV::OpenSaveDataFileSystem(Kernel::HLERequestContext& ctx) { | |||
| 857 | case FileSys::SaveDataSpaceId::NandSystem: | 856 | case FileSys::SaveDataSpaceId::NandSystem: |
| 858 | id = FileSys::StorageId::NandSystem; | 857 | id = FileSys::StorageId::NandSystem; |
| 859 | break; | 858 | break; |
| 859 | case FileSys::SaveDataSpaceId::TemporaryStorage: | ||
| 860 | case FileSys::SaveDataSpaceId::ProperSystem: | ||
| 861 | case FileSys::SaveDataSpaceId::SafeMode: | ||
| 862 | UNREACHABLE(); | ||
| 860 | } | 863 | } |
| 861 | 864 | ||
| 862 | auto filesystem = | 865 | auto filesystem = |
| @@ -902,7 +905,14 @@ void FSP_SRV::ReadSaveDataFileSystemExtraDataWithMaskBySaveDataAttribute( | |||
| 902 | // Stub this to None for now, backend needs an impl to read/write the SaveDataExtraData | 905 | // Stub this to None for now, backend needs an impl to read/write the SaveDataExtraData |
| 903 | constexpr auto flags = static_cast<u32>(FileSys::SaveDataFlags::None); | 906 | constexpr auto flags = static_cast<u32>(FileSys::SaveDataFlags::None); |
| 904 | 907 | ||
| 905 | LOG_WARNING(Service_FS, "(STUBBED) called, flags={}", flags); | 908 | LOG_WARNING(Service_FS, |
| 909 | "(STUBBED) called, flags={}, space_id={}, attribute.title_id={:016X}\n" | ||
| 910 | "attribute.user_id={:016X}{:016X}, attribute.save_id={:016X}\n" | ||
| 911 | "attribute.type={}, attribute.rank={}, attribute.index={}", | ||
| 912 | flags, static_cast<u32>(parameters.space_id), parameters.attribute.title_id, | ||
| 913 | parameters.attribute.user_id[1], parameters.attribute.user_id[0], | ||
| 914 | parameters.attribute.save_id, static_cast<u32>(parameters.attribute.type), | ||
| 915 | static_cast<u32>(parameters.attribute.rank), parameters.attribute.index); | ||
| 906 | 916 | ||
| 907 | IPC::ResponseBuilder rb{ctx, 3}; | 917 | IPC::ResponseBuilder rb{ctx, 3}; |
| 908 | rb.Push(RESULT_SUCCESS); | 918 | rb.Push(RESULT_SUCCESS); |
diff --git a/src/core/hle/service/hid/controllers/npad.cpp b/src/core/hle/service/hid/controllers/npad.cpp index 45fde8df2..e742497e1 100644 --- a/src/core/hle/service/hid/controllers/npad.cpp +++ b/src/core/hle/service/hid/controllers/npad.cpp | |||
| @@ -574,6 +574,22 @@ Controller_NPad::GyroscopeZeroDriftMode Controller_NPad::GetGyroscopeZeroDriftMo | |||
| 574 | return gyroscope_zero_drift_mode; | 574 | return gyroscope_zero_drift_mode; |
| 575 | } | 575 | } |
| 576 | 576 | ||
| 577 | void Controller_NPad::MergeSingleJoyAsDualJoy(u32 npad_id_1, u32 npad_id_2) { | ||
| 578 | const auto npad_index_1 = NPadIdToIndex(npad_id_1); | ||
| 579 | const auto npad_index_2 = NPadIdToIndex(npad_id_2); | ||
| 580 | |||
| 581 | // If the controllers at both npad indices form a pair of left and right joycons, merge them. | ||
| 582 | // Otherwise, do nothing. | ||
| 583 | if ((connected_controllers[npad_index_1].type == NPadControllerType::JoyLeft && | ||
| 584 | connected_controllers[npad_index_2].type == NPadControllerType::JoyRight) || | ||
| 585 | (connected_controllers[npad_index_2].type == NPadControllerType::JoyLeft && | ||
| 586 | connected_controllers[npad_index_1].type == NPadControllerType::JoyRight)) { | ||
| 587 | // Disconnect the joycon at the second id and connect the dual joycon at the first index. | ||
| 588 | DisconnectNPad(npad_id_2); | ||
| 589 | AddNewControllerAt(NPadControllerType::JoyDual, npad_index_1); | ||
| 590 | } | ||
| 591 | } | ||
| 592 | |||
| 577 | void Controller_NPad::StartLRAssignmentMode() { | 593 | void Controller_NPad::StartLRAssignmentMode() { |
| 578 | // Nothing internally is used for lr assignment mode. Since we have the ability to set the | 594 | // Nothing internally is used for lr assignment mode. Since we have the ability to set the |
| 579 | // controller types from boot, it doesn't really matter about showing a selection screen | 595 | // controller types from boot, it doesn't really matter about showing a selection screen |
diff --git a/src/core/hle/service/hid/controllers/npad.h b/src/core/hle/service/hid/controllers/npad.h index 75ce5b731..ad25c6fbf 100644 --- a/src/core/hle/service/hid/controllers/npad.h +++ b/src/core/hle/service/hid/controllers/npad.h | |||
| @@ -134,6 +134,7 @@ public: | |||
| 134 | void ConnectAllDisconnectedControllers(); | 134 | void ConnectAllDisconnectedControllers(); |
| 135 | void ClearAllControllers(); | 135 | void ClearAllControllers(); |
| 136 | 136 | ||
| 137 | void MergeSingleJoyAsDualJoy(u32 npad_id_1, u32 npad_id_2); | ||
| 137 | void StartLRAssignmentMode(); | 138 | void StartLRAssignmentMode(); |
| 138 | void StopLRAssignmentMode(); | 139 | void StopLRAssignmentMode(); |
| 139 | bool SwapNpadAssignment(u32 npad_id_1, u32 npad_id_2); | 140 | bool SwapNpadAssignment(u32 npad_id_1, u32 npad_id_2); |
diff --git a/src/core/hle/service/hid/controllers/touchscreen.cpp b/src/core/hle/service/hid/controllers/touchscreen.cpp index e326f8f5c..0df395e85 100644 --- a/src/core/hle/service/hid/controllers/touchscreen.cpp +++ b/src/core/hle/service/hid/controllers/touchscreen.cpp | |||
| @@ -40,9 +40,14 @@ void Controller_Touchscreen::OnUpdate(const Core::Timing::CoreTiming& core_timin | |||
| 40 | cur_entry.sampling_number = last_entry.sampling_number + 1; | 40 | cur_entry.sampling_number = last_entry.sampling_number + 1; |
| 41 | cur_entry.sampling_number2 = cur_entry.sampling_number; | 41 | cur_entry.sampling_number2 = cur_entry.sampling_number; |
| 42 | 42 | ||
| 43 | const auto [x, y, pressed] = touch_device->GetStatus(); | 43 | bool pressed = false; |
| 44 | float x, y; | ||
| 45 | std::tie(x, y, pressed) = touch_device->GetStatus(); | ||
| 44 | auto& touch_entry = cur_entry.states[0]; | 46 | auto& touch_entry = cur_entry.states[0]; |
| 45 | touch_entry.attribute.raw = 0; | 47 | touch_entry.attribute.raw = 0; |
| 48 | if (!pressed && touch_btn_device) { | ||
| 49 | std::tie(x, y, pressed) = touch_btn_device->GetStatus(); | ||
| 50 | } | ||
| 46 | if (pressed && Settings::values.touchscreen.enabled) { | 51 | if (pressed && Settings::values.touchscreen.enabled) { |
| 47 | touch_entry.x = static_cast<u16>(x * Layout::ScreenUndocked::Width); | 52 | touch_entry.x = static_cast<u16>(x * Layout::ScreenUndocked::Width); |
| 48 | touch_entry.y = static_cast<u16>(y * Layout::ScreenUndocked::Height); | 53 | touch_entry.y = static_cast<u16>(y * Layout::ScreenUndocked::Height); |
| @@ -63,5 +68,10 @@ void Controller_Touchscreen::OnUpdate(const Core::Timing::CoreTiming& core_timin | |||
| 63 | 68 | ||
| 64 | void Controller_Touchscreen::OnLoadInputDevices() { | 69 | void Controller_Touchscreen::OnLoadInputDevices() { |
| 65 | touch_device = Input::CreateDevice<Input::TouchDevice>(Settings::values.touchscreen.device); | 70 | touch_device = Input::CreateDevice<Input::TouchDevice>(Settings::values.touchscreen.device); |
| 71 | if (Settings::values.use_touch_from_button) { | ||
| 72 | touch_btn_device = Input::CreateDevice<Input::TouchDevice>("engine:touch_from_button"); | ||
| 73 | } else { | ||
| 74 | touch_btn_device.reset(); | ||
| 75 | } | ||
| 66 | } | 76 | } |
| 67 | } // namespace Service::HID | 77 | } // namespace Service::HID |
diff --git a/src/core/hle/service/hid/controllers/touchscreen.h b/src/core/hle/service/hid/controllers/touchscreen.h index a1d97269e..4d9042adc 100644 --- a/src/core/hle/service/hid/controllers/touchscreen.h +++ b/src/core/hle/service/hid/controllers/touchscreen.h | |||
| @@ -68,6 +68,7 @@ private: | |||
| 68 | "TouchScreenSharedMemory is an invalid size"); | 68 | "TouchScreenSharedMemory is an invalid size"); |
| 69 | TouchScreenSharedMemory shared_memory{}; | 69 | TouchScreenSharedMemory shared_memory{}; |
| 70 | std::unique_ptr<Input::TouchDevice> touch_device; | 70 | std::unique_ptr<Input::TouchDevice> touch_device; |
| 71 | std::unique_ptr<Input::TouchDevice> touch_btn_device; | ||
| 71 | s64_le last_touch{}; | 72 | s64_le last_touch{}; |
| 72 | }; | 73 | }; |
| 73 | } // namespace Service::HID | 74 | } // namespace Service::HID |
diff --git a/src/core/hle/service/hid/hid.cpp b/src/core/hle/service/hid/hid.cpp index 33416b5dd..bd3c2f26b 100644 --- a/src/core/hle/service/hid/hid.cpp +++ b/src/core/hle/service/hid/hid.cpp | |||
| @@ -671,13 +671,15 @@ void Hid::SetNpadJoyAssignmentModeDual(Kernel::HLERequestContext& ctx) { | |||
| 671 | 671 | ||
| 672 | void Hid::MergeSingleJoyAsDualJoy(Kernel::HLERequestContext& ctx) { | 672 | void Hid::MergeSingleJoyAsDualJoy(Kernel::HLERequestContext& ctx) { |
| 673 | IPC::RequestParser rp{ctx}; | 673 | IPC::RequestParser rp{ctx}; |
| 674 | const auto unknown_1{rp.Pop<u32>()}; | 674 | const auto npad_id_1{rp.Pop<u32>()}; |
| 675 | const auto unknown_2{rp.Pop<u32>()}; | 675 | const auto npad_id_2{rp.Pop<u32>()}; |
| 676 | const auto applet_resource_user_id{rp.Pop<u64>()}; | 676 | const auto applet_resource_user_id{rp.Pop<u64>()}; |
| 677 | 677 | ||
| 678 | LOG_WARNING(Service_HID, | 678 | LOG_DEBUG(Service_HID, "called, npad_id_1={}, npad_id_2={}, applet_resource_user_id={}", |
| 679 | "(STUBBED) called, unknown_1={}, unknown_2={}, applet_resource_user_id={}", | 679 | npad_id_1, npad_id_2, applet_resource_user_id); |
| 680 | unknown_1, unknown_2, applet_resource_user_id); | 680 | |
| 681 | auto& controller = applet_resource->GetController<Controller_NPad>(HidController::NPad); | ||
| 682 | controller.MergeSingleJoyAsDualJoy(npad_id_1, npad_id_2); | ||
| 681 | 683 | ||
| 682 | IPC::ResponseBuilder rb{ctx, 2}; | 684 | IPC::ResponseBuilder rb{ctx, 2}; |
| 683 | rb.Push(RESULT_SUCCESS); | 685 | rb.Push(RESULT_SUCCESS); |
diff --git a/src/core/hle/service/nifm/nifm.cpp b/src/core/hle/service/nifm/nifm.cpp index 01ddcdbd6..2e9d95195 100644 --- a/src/core/hle/service/nifm/nifm.cpp +++ b/src/core/hle/service/nifm/nifm.cpp | |||
| @@ -9,6 +9,7 @@ | |||
| 9 | #include "core/hle/kernel/writable_event.h" | 9 | #include "core/hle/kernel/writable_event.h" |
| 10 | #include "core/hle/service/nifm/nifm.h" | 10 | #include "core/hle/service/nifm/nifm.h" |
| 11 | #include "core/hle/service/service.h" | 11 | #include "core/hle/service/service.h" |
| 12 | #include "core/network/network.h" | ||
| 12 | #include "core/settings.h" | 13 | #include "core/settings.h" |
| 13 | 14 | ||
| 14 | namespace Service::NIFM { | 15 | namespace Service::NIFM { |
| @@ -174,6 +175,16 @@ private: | |||
| 174 | IPC::ResponseBuilder rb{ctx, 2}; | 175 | IPC::ResponseBuilder rb{ctx, 2}; |
| 175 | rb.Push(RESULT_SUCCESS); | 176 | rb.Push(RESULT_SUCCESS); |
| 176 | } | 177 | } |
| 178 | void GetCurrentIpAddress(Kernel::HLERequestContext& ctx) { | ||
| 179 | LOG_WARNING(Service_NIFM, "(STUBBED) called"); | ||
| 180 | |||
| 181 | const auto [ipv4, error] = Network::GetHostIPv4Address(); | ||
| 182 | UNIMPLEMENTED_IF(error != Network::Errno::SUCCESS); | ||
| 183 | |||
| 184 | IPC::ResponseBuilder rb{ctx, 3}; | ||
| 185 | rb.Push(RESULT_SUCCESS); | ||
| 186 | rb.PushRaw(ipv4); | ||
| 187 | } | ||
| 177 | void CreateTemporaryNetworkProfile(Kernel::HLERequestContext& ctx) { | 188 | void CreateTemporaryNetworkProfile(Kernel::HLERequestContext& ctx) { |
| 178 | LOG_DEBUG(Service_NIFM, "called"); | 189 | LOG_DEBUG(Service_NIFM, "called"); |
| 179 | 190 | ||
| @@ -235,7 +246,7 @@ IGeneralService::IGeneralService(Core::System& system) | |||
| 235 | {9, nullptr, "SetNetworkProfile"}, | 246 | {9, nullptr, "SetNetworkProfile"}, |
| 236 | {10, &IGeneralService::RemoveNetworkProfile, "RemoveNetworkProfile"}, | 247 | {10, &IGeneralService::RemoveNetworkProfile, "RemoveNetworkProfile"}, |
| 237 | {11, nullptr, "GetScanDataOld"}, | 248 | {11, nullptr, "GetScanDataOld"}, |
| 238 | {12, nullptr, "GetCurrentIpAddress"}, | 249 | {12, &IGeneralService::GetCurrentIpAddress, "GetCurrentIpAddress"}, |
| 239 | {13, nullptr, "GetCurrentAccessPointOld"}, | 250 | {13, nullptr, "GetCurrentAccessPointOld"}, |
| 240 | {14, &IGeneralService::CreateTemporaryNetworkProfile, "CreateTemporaryNetworkProfile"}, | 251 | {14, &IGeneralService::CreateTemporaryNetworkProfile, "CreateTemporaryNetworkProfile"}, |
| 241 | {15, nullptr, "GetCurrentIpConfigInfo"}, | 252 | {15, nullptr, "GetCurrentIpConfigInfo"}, |
diff --git a/src/core/hle/service/ns/ns.cpp b/src/core/hle/service/ns/ns.cpp index 886450be2..58ee1f712 100644 --- a/src/core/hle/service/ns/ns.cpp +++ b/src/core/hle/service/ns/ns.cpp | |||
| @@ -5,6 +5,7 @@ | |||
| 5 | #include "common/logging/log.h" | 5 | #include "common/logging/log.h" |
| 6 | #include "core/file_sys/control_metadata.h" | 6 | #include "core/file_sys/control_metadata.h" |
| 7 | #include "core/file_sys/patch_manager.h" | 7 | #include "core/file_sys/patch_manager.h" |
| 8 | #include "core/file_sys/vfs.h" | ||
| 8 | #include "core/hle/ipc_helpers.h" | 9 | #include "core/hle/ipc_helpers.h" |
| 9 | #include "core/hle/kernel/hle_ipc.h" | 10 | #include "core/hle/kernel/hle_ipc.h" |
| 10 | #include "core/hle/service/ns/errors.h" | 11 | #include "core/hle/service/ns/errors.h" |
diff --git a/src/core/hle/service/service.cpp b/src/core/hle/service/service.cpp index fa5347af9..538f28495 100644 --- a/src/core/hle/service/service.cpp +++ b/src/core/hle/service/service.cpp | |||
| @@ -246,7 +246,7 @@ void Init(std::shared_ptr<SM::ServiceManager>& sm, Core::System& system) { | |||
| 246 | PSC::InstallInterfaces(*sm); | 246 | PSC::InstallInterfaces(*sm); |
| 247 | PSM::InstallInterfaces(*sm); | 247 | PSM::InstallInterfaces(*sm); |
| 248 | Set::InstallInterfaces(*sm); | 248 | Set::InstallInterfaces(*sm); |
| 249 | Sockets::InstallInterfaces(*sm); | 249 | Sockets::InstallInterfaces(*sm, system); |
| 250 | SPL::InstallInterfaces(*sm); | 250 | SPL::InstallInterfaces(*sm); |
| 251 | SSL::InstallInterfaces(*sm); | 251 | SSL::InstallInterfaces(*sm); |
| 252 | Time::InstallInterfaces(system); | 252 | Time::InstallInterfaces(system); |
diff --git a/src/core/hle/service/sockets/blocking_worker.h b/src/core/hle/service/sockets/blocking_worker.h new file mode 100644 index 000000000..31ef6b821 --- /dev/null +++ b/src/core/hle/service/sockets/blocking_worker.h | |||
| @@ -0,0 +1,162 @@ | |||
| 1 | // Copyright 2020 yuzu emulator team | ||
| 2 | // Licensed under GPLv2 or any later version | ||
| 3 | // Refer to the license.txt file included. | ||
| 4 | |||
| 5 | #pragma once | ||
| 6 | |||
| 7 | #include <atomic> | ||
| 8 | #include <memory> | ||
| 9 | #include <string> | ||
| 10 | #include <string_view> | ||
| 11 | #include <thread> | ||
| 12 | #include <variant> | ||
| 13 | #include <vector> | ||
| 14 | |||
| 15 | #include <fmt/format.h> | ||
| 16 | |||
| 17 | #include "common/assert.h" | ||
| 18 | #include "common/microprofile.h" | ||
| 19 | #include "common/thread.h" | ||
| 20 | #include "core/core.h" | ||
| 21 | #include "core/hle/kernel/hle_ipc.h" | ||
| 22 | #include "core/hle/kernel/kernel.h" | ||
| 23 | #include "core/hle/kernel/thread.h" | ||
| 24 | #include "core/hle/kernel/writable_event.h" | ||
| 25 | |||
| 26 | namespace Service::Sockets { | ||
| 27 | |||
| 28 | /** | ||
| 29 | * Worker abstraction to execute blocking calls on host without blocking the guest thread | ||
| 30 | * | ||
| 31 | * @tparam Service Service where the work is executed | ||
| 32 | * @tparam ...Types Types of work to execute | ||
| 33 | */ | ||
| 34 | template <class Service, class... Types> | ||
| 35 | class BlockingWorker { | ||
| 36 | using This = BlockingWorker<Service, Types...>; | ||
| 37 | using WorkVariant = std::variant<std::monostate, Types...>; | ||
| 38 | |||
| 39 | public: | ||
| 40 | /// Create a new worker | ||
| 41 | static std::unique_ptr<This> Create(Core::System& system, Service* service, | ||
| 42 | std::string_view name) { | ||
| 43 | return std::unique_ptr<This>(new This(system, service, name)); | ||
| 44 | } | ||
| 45 | |||
| 46 | ~BlockingWorker() { | ||
| 47 | while (!is_available.load(std::memory_order_relaxed)) { | ||
| 48 | // Busy wait until work is finished | ||
| 49 | std::this_thread::yield(); | ||
| 50 | } | ||
| 51 | // Monostate means to exit the thread | ||
| 52 | work = std::monostate{}; | ||
| 53 | work_event.Set(); | ||
| 54 | thread.join(); | ||
| 55 | } | ||
| 56 | |||
| 57 | /** | ||
| 58 | * Try to capture the worker to send work after a success | ||
| 59 | * @returns True when the worker has been successfully captured | ||
| 60 | */ | ||
| 61 | bool TryCapture() { | ||
| 62 | bool expected = true; | ||
| 63 | return is_available.compare_exchange_weak(expected, false, std::memory_order_relaxed, | ||
| 64 | std::memory_order_relaxed); | ||
| 65 | } | ||
| 66 | |||
| 67 | /** | ||
| 68 | * Send work to this worker abstraction | ||
| 69 | * @see TryCapture must be called before attempting to call this function | ||
| 70 | */ | ||
| 71 | template <class Work> | ||
| 72 | void SendWork(Work new_work) { | ||
| 73 | ASSERT_MSG(!is_available, "Trying to send work on a worker that's not captured"); | ||
| 74 | work = std::move(new_work); | ||
| 75 | work_event.Set(); | ||
| 76 | } | ||
| 77 | |||
| 78 | /// Generate a callback for @see SleepClientThread | ||
| 79 | template <class Work> | ||
| 80 | auto Callback() { | ||
| 81 | return [this](std::shared_ptr<Kernel::Thread>, Kernel::HLERequestContext& ctx, | ||
| 82 | Kernel::ThreadWakeupReason reason) { | ||
| 83 | ASSERT(reason == Kernel::ThreadWakeupReason::Signal); | ||
| 84 | std::get<Work>(work).Response(ctx); | ||
| 85 | is_available.store(true); | ||
| 86 | }; | ||
| 87 | } | ||
| 88 | |||
| 89 | /// Get kernel event that will be signalled by the worker when the host operation finishes | ||
| 90 | std::shared_ptr<Kernel::WritableEvent> KernelEvent() const { | ||
| 91 | return kernel_event; | ||
| 92 | } | ||
| 93 | |||
| 94 | private: | ||
| 95 | explicit BlockingWorker(Core::System& system, Service* service, std::string_view name) { | ||
| 96 | auto pair = Kernel::WritableEvent::CreateEventPair(system.Kernel(), std::string(name)); | ||
| 97 | kernel_event = std::move(pair.writable); | ||
| 98 | thread = std::thread([this, &system, service, name] { Run(system, service, name); }); | ||
| 99 | } | ||
| 100 | |||
| 101 | void Run(Core::System& system, Service* service, std::string_view name) { | ||
| 102 | system.RegisterHostThread(); | ||
| 103 | |||
| 104 | const std::string thread_name = fmt::format("yuzu:{}", name); | ||
| 105 | MicroProfileOnThreadCreate(thread_name.c_str()); | ||
| 106 | Common::SetCurrentThreadName(thread_name.c_str()); | ||
| 107 | |||
| 108 | bool keep_running = true; | ||
| 109 | while (keep_running) { | ||
| 110 | work_event.Wait(); | ||
| 111 | |||
| 112 | const auto visit_fn = [service, &keep_running](auto&& w) { | ||
| 113 | using T = std::decay_t<decltype(w)>; | ||
| 114 | if constexpr (std::is_same_v<T, std::monostate>) { | ||
| 115 | keep_running = false; | ||
| 116 | } else { | ||
| 117 | w.Execute(service); | ||
| 118 | } | ||
| 119 | }; | ||
| 120 | std::visit(visit_fn, work); | ||
| 121 | |||
| 122 | kernel_event->Signal(); | ||
| 123 | } | ||
| 124 | } | ||
| 125 | |||
| 126 | std::thread thread; | ||
| 127 | WorkVariant work; | ||
| 128 | Common::Event work_event; | ||
| 129 | std::shared_ptr<Kernel::WritableEvent> kernel_event; | ||
| 130 | std::atomic_bool is_available{true}; | ||
| 131 | }; | ||
| 132 | |||
| 133 | template <class Service, class... Types> | ||
| 134 | class BlockingWorkerPool { | ||
| 135 | using Worker = BlockingWorker<Service, Types...>; | ||
| 136 | |||
| 137 | public: | ||
| 138 | explicit BlockingWorkerPool(Core::System& system_, Service* service_) | ||
| 139 | : system{system_}, service{service_} {} | ||
| 140 | |||
| 141 | /// Returns a captured worker thread, creating new ones if necessary | ||
| 142 | Worker* CaptureWorker() { | ||
| 143 | for (auto& worker : workers) { | ||
| 144 | if (worker->TryCapture()) { | ||
| 145 | return worker.get(); | ||
| 146 | } | ||
| 147 | } | ||
| 148 | auto new_worker = Worker::Create(system, service, fmt::format("BSD:{}", workers.size())); | ||
| 149 | [[maybe_unused]] const bool success = new_worker->TryCapture(); | ||
| 150 | ASSERT(success); | ||
| 151 | |||
| 152 | return workers.emplace_back(std::move(new_worker)).get(); | ||
| 153 | } | ||
| 154 | |||
| 155 | private: | ||
| 156 | Core::System& system; | ||
| 157 | Service* const service; | ||
| 158 | |||
| 159 | std::vector<std::unique_ptr<Worker>> workers; | ||
| 160 | }; | ||
| 161 | |||
| 162 | } // namespace Service::Sockets | ||
diff --git a/src/core/hle/service/sockets/bsd.cpp b/src/core/hle/service/sockets/bsd.cpp index 8d4952c0e..803505452 100644 --- a/src/core/hle/service/sockets/bsd.cpp +++ b/src/core/hle/service/sockets/bsd.cpp | |||
| @@ -2,18 +2,138 @@ | |||
| 2 | // Licensed under GPLv2 or any later version | 2 | // Licensed under GPLv2 or any later version |
| 3 | // Refer to the license.txt file included. | 3 | // Refer to the license.txt file included. |
| 4 | 4 | ||
| 5 | #include <array> | ||
| 6 | #include <memory> | ||
| 7 | #include <string> | ||
| 8 | #include <utility> | ||
| 9 | #include <vector> | ||
| 10 | |||
| 11 | #include <fmt/format.h> | ||
| 12 | |||
| 13 | #include "common/microprofile.h" | ||
| 14 | #include "common/thread.h" | ||
| 5 | #include "core/hle/ipc_helpers.h" | 15 | #include "core/hle/ipc_helpers.h" |
| 16 | #include "core/hle/kernel/thread.h" | ||
| 6 | #include "core/hle/service/sockets/bsd.h" | 17 | #include "core/hle/service/sockets/bsd.h" |
| 18 | #include "core/hle/service/sockets/sockets_translate.h" | ||
| 19 | #include "core/network/network.h" | ||
| 20 | #include "core/network/sockets.h" | ||
| 7 | 21 | ||
| 8 | namespace Service::Sockets { | 22 | namespace Service::Sockets { |
| 9 | 23 | ||
| 24 | namespace { | ||
| 25 | |||
| 26 | bool IsConnectionBased(Type type) { | ||
| 27 | switch (type) { | ||
| 28 | case Type::STREAM: | ||
| 29 | return true; | ||
| 30 | case Type::DGRAM: | ||
| 31 | return false; | ||
| 32 | default: | ||
| 33 | UNIMPLEMENTED_MSG("Unimplemented type={}", static_cast<int>(type)); | ||
| 34 | return false; | ||
| 35 | } | ||
| 36 | } | ||
| 37 | |||
| 38 | } // Anonymous namespace | ||
| 39 | |||
| 40 | void BSD::PollWork::Execute(BSD* bsd) { | ||
| 41 | std::tie(ret, bsd_errno) = bsd->PollImpl(write_buffer, read_buffer, nfds, timeout); | ||
| 42 | } | ||
| 43 | |||
| 44 | void BSD::PollWork::Response(Kernel::HLERequestContext& ctx) { | ||
| 45 | ctx.WriteBuffer(write_buffer); | ||
| 46 | |||
| 47 | IPC::ResponseBuilder rb{ctx, 4}; | ||
| 48 | rb.Push(RESULT_SUCCESS); | ||
| 49 | rb.Push<s32>(ret); | ||
| 50 | rb.PushEnum(bsd_errno); | ||
| 51 | } | ||
| 52 | |||
| 53 | void BSD::AcceptWork::Execute(BSD* bsd) { | ||
| 54 | std::tie(ret, bsd_errno) = bsd->AcceptImpl(fd, write_buffer); | ||
| 55 | } | ||
| 56 | |||
| 57 | void BSD::AcceptWork::Response(Kernel::HLERequestContext& ctx) { | ||
| 58 | ctx.WriteBuffer(write_buffer); | ||
| 59 | |||
| 60 | IPC::ResponseBuilder rb{ctx, 5}; | ||
| 61 | rb.Push(RESULT_SUCCESS); | ||
| 62 | rb.Push<s32>(ret); | ||
| 63 | rb.PushEnum(bsd_errno); | ||
| 64 | rb.Push<u32>(static_cast<u32>(write_buffer.size())); | ||
| 65 | } | ||
| 66 | |||
| 67 | void BSD::ConnectWork::Execute(BSD* bsd) { | ||
| 68 | bsd_errno = bsd->ConnectImpl(fd, addr); | ||
| 69 | } | ||
| 70 | |||
| 71 | void BSD::ConnectWork::Response(Kernel::HLERequestContext& ctx) { | ||
| 72 | IPC::ResponseBuilder rb{ctx, 4}; | ||
| 73 | rb.Push(RESULT_SUCCESS); | ||
| 74 | rb.Push<s32>(bsd_errno == Errno::SUCCESS ? 0 : -1); | ||
| 75 | rb.PushEnum(bsd_errno); | ||
| 76 | } | ||
| 77 | |||
| 78 | void BSD::RecvWork::Execute(BSD* bsd) { | ||
| 79 | std::tie(ret, bsd_errno) = bsd->RecvImpl(fd, flags, message); | ||
| 80 | } | ||
| 81 | |||
| 82 | void BSD::RecvWork::Response(Kernel::HLERequestContext& ctx) { | ||
| 83 | ctx.WriteBuffer(message); | ||
| 84 | |||
| 85 | IPC::ResponseBuilder rb{ctx, 4}; | ||
| 86 | rb.Push(RESULT_SUCCESS); | ||
| 87 | rb.Push<s32>(ret); | ||
| 88 | rb.PushEnum(bsd_errno); | ||
| 89 | } | ||
| 90 | |||
| 91 | void BSD::RecvFromWork::Execute(BSD* bsd) { | ||
| 92 | std::tie(ret, bsd_errno) = bsd->RecvFromImpl(fd, flags, message, addr); | ||
| 93 | } | ||
| 94 | |||
| 95 | void BSD::RecvFromWork::Response(Kernel::HLERequestContext& ctx) { | ||
| 96 | ctx.WriteBuffer(message, 0); | ||
| 97 | if (!addr.empty()) { | ||
| 98 | ctx.WriteBuffer(addr, 1); | ||
| 99 | } | ||
| 100 | |||
| 101 | IPC::ResponseBuilder rb{ctx, 5}; | ||
| 102 | rb.Push(RESULT_SUCCESS); | ||
| 103 | rb.Push<s32>(ret); | ||
| 104 | rb.PushEnum(bsd_errno); | ||
| 105 | rb.Push<u32>(static_cast<u32>(addr.size())); | ||
| 106 | } | ||
| 107 | |||
| 108 | void BSD::SendWork::Execute(BSD* bsd) { | ||
| 109 | std::tie(ret, bsd_errno) = bsd->SendImpl(fd, flags, message); | ||
| 110 | } | ||
| 111 | |||
| 112 | void BSD::SendWork::Response(Kernel::HLERequestContext& ctx) { | ||
| 113 | IPC::ResponseBuilder rb{ctx, 4}; | ||
| 114 | rb.Push(RESULT_SUCCESS); | ||
| 115 | rb.Push<s32>(ret); | ||
| 116 | rb.PushEnum(bsd_errno); | ||
| 117 | } | ||
| 118 | |||
| 119 | void BSD::SendToWork::Execute(BSD* bsd) { | ||
| 120 | std::tie(ret, bsd_errno) = bsd->SendToImpl(fd, flags, message, addr); | ||
| 121 | } | ||
| 122 | |||
| 123 | void BSD::SendToWork::Response(Kernel::HLERequestContext& ctx) { | ||
| 124 | IPC::ResponseBuilder rb{ctx, 4}; | ||
| 125 | rb.Push(RESULT_SUCCESS); | ||
| 126 | rb.Push<s32>(ret); | ||
| 127 | rb.PushEnum(bsd_errno); | ||
| 128 | } | ||
| 129 | |||
| 10 | void BSD::RegisterClient(Kernel::HLERequestContext& ctx) { | 130 | void BSD::RegisterClient(Kernel::HLERequestContext& ctx) { |
| 11 | LOG_WARNING(Service, "(STUBBED) called"); | 131 | LOG_WARNING(Service, "(STUBBED) called"); |
| 12 | 132 | ||
| 13 | IPC::ResponseBuilder rb{ctx, 3}; | 133 | IPC::ResponseBuilder rb{ctx, 3}; |
| 14 | 134 | ||
| 15 | rb.Push(RESULT_SUCCESS); | 135 | rb.Push(RESULT_SUCCESS); |
| 16 | rb.Push<u32>(0); // bsd errno | 136 | rb.Push<s32>(0); // bsd errno |
| 17 | } | 137 | } |
| 18 | 138 | ||
| 19 | void BSD::StartMonitoring(Kernel::HLERequestContext& ctx) { | 139 | void BSD::StartMonitoring(Kernel::HLERequestContext& ctx) { |
| @@ -26,20 +146,19 @@ void BSD::StartMonitoring(Kernel::HLERequestContext& ctx) { | |||
| 26 | 146 | ||
| 27 | void BSD::Socket(Kernel::HLERequestContext& ctx) { | 147 | void BSD::Socket(Kernel::HLERequestContext& ctx) { |
| 28 | IPC::RequestParser rp{ctx}; | 148 | IPC::RequestParser rp{ctx}; |
| 149 | const u32 domain = rp.Pop<u32>(); | ||
| 150 | const u32 type = rp.Pop<u32>(); | ||
| 151 | const u32 protocol = rp.Pop<u32>(); | ||
| 29 | 152 | ||
| 30 | u32 domain = rp.Pop<u32>(); | 153 | LOG_DEBUG(Service, "called. domain={} type={} protocol={}", domain, type, protocol); |
| 31 | u32 type = rp.Pop<u32>(); | ||
| 32 | u32 protocol = rp.Pop<u32>(); | ||
| 33 | |||
| 34 | LOG_WARNING(Service, "(STUBBED) called domain={} type={} protocol={}", domain, type, protocol); | ||
| 35 | 154 | ||
| 36 | u32 fd = next_fd++; | 155 | const auto [fd, bsd_errno] = SocketImpl(static_cast<Domain>(domain), static_cast<Type>(type), |
| 156 | static_cast<Protocol>(protocol)); | ||
| 37 | 157 | ||
| 38 | IPC::ResponseBuilder rb{ctx, 4}; | 158 | IPC::ResponseBuilder rb{ctx, 4}; |
| 39 | |||
| 40 | rb.Push(RESULT_SUCCESS); | 159 | rb.Push(RESULT_SUCCESS); |
| 41 | rb.Push<u32>(fd); | 160 | rb.Push<s32>(fd); |
| 42 | rb.Push<u32>(0); // bsd errno | 161 | rb.PushEnum(bsd_errno); |
| 43 | } | 162 | } |
| 44 | 163 | ||
| 45 | void BSD::Select(Kernel::HLERequestContext& ctx) { | 164 | void BSD::Select(Kernel::HLERequestContext& ctx) { |
| @@ -52,67 +171,663 @@ void BSD::Select(Kernel::HLERequestContext& ctx) { | |||
| 52 | rb.Push<u32>(0); // bsd errno | 171 | rb.Push<u32>(0); // bsd errno |
| 53 | } | 172 | } |
| 54 | 173 | ||
| 174 | void BSD::Poll(Kernel::HLERequestContext& ctx) { | ||
| 175 | IPC::RequestParser rp{ctx}; | ||
| 176 | const s32 nfds = rp.Pop<s32>(); | ||
| 177 | const s32 timeout = rp.Pop<s32>(); | ||
| 178 | |||
| 179 | LOG_DEBUG(Service, "called. nfds={} timeout={}", nfds, timeout); | ||
| 180 | |||
| 181 | ExecuteWork(ctx, "BSD:Poll", timeout != 0, | ||
| 182 | PollWork{ | ||
| 183 | .nfds = nfds, | ||
| 184 | .timeout = timeout, | ||
| 185 | .read_buffer = ctx.ReadBuffer(), | ||
| 186 | .write_buffer = std::vector<u8>(ctx.GetWriteBufferSize()), | ||
| 187 | }); | ||
| 188 | } | ||
| 189 | |||
| 190 | void BSD::Accept(Kernel::HLERequestContext& ctx) { | ||
| 191 | IPC::RequestParser rp{ctx}; | ||
| 192 | const s32 fd = rp.Pop<s32>(); | ||
| 193 | |||
| 194 | LOG_DEBUG(Service, "called. fd={}", fd); | ||
| 195 | |||
| 196 | ExecuteWork(ctx, "BSD:Accept", IsBlockingSocket(fd), | ||
| 197 | AcceptWork{ | ||
| 198 | .fd = fd, | ||
| 199 | .write_buffer = std::vector<u8>(ctx.GetWriteBufferSize()), | ||
| 200 | }); | ||
| 201 | } | ||
| 202 | |||
| 55 | void BSD::Bind(Kernel::HLERequestContext& ctx) { | 203 | void BSD::Bind(Kernel::HLERequestContext& ctx) { |
| 56 | LOG_WARNING(Service, "(STUBBED) called"); | 204 | IPC::RequestParser rp{ctx}; |
| 205 | const s32 fd = rp.Pop<s32>(); | ||
| 57 | 206 | ||
| 58 | IPC::ResponseBuilder rb{ctx, 4}; | 207 | LOG_DEBUG(Service, "called. fd={} addrlen={}", fd, ctx.GetReadBufferSize()); |
| 59 | 208 | ||
| 60 | rb.Push(RESULT_SUCCESS); | 209 | BuildErrnoResponse(ctx, BindImpl(fd, ctx.ReadBuffer())); |
| 61 | rb.Push<u32>(0); // ret | ||
| 62 | rb.Push<u32>(0); // bsd errno | ||
| 63 | } | 210 | } |
| 64 | 211 | ||
| 65 | void BSD::Connect(Kernel::HLERequestContext& ctx) { | 212 | void BSD::Connect(Kernel::HLERequestContext& ctx) { |
| 66 | LOG_WARNING(Service, "(STUBBED) called"); | 213 | IPC::RequestParser rp{ctx}; |
| 214 | const s32 fd = rp.Pop<s32>(); | ||
| 67 | 215 | ||
| 68 | IPC::ResponseBuilder rb{ctx, 4}; | 216 | LOG_DEBUG(Service, "called. fd={} addrlen={}", fd, ctx.GetReadBufferSize()); |
| 217 | |||
| 218 | ExecuteWork(ctx, "BSD:Connect", IsBlockingSocket(fd), | ||
| 219 | ConnectWork{ | ||
| 220 | .fd = fd, | ||
| 221 | .addr = ctx.ReadBuffer(), | ||
| 222 | }); | ||
| 223 | } | ||
| 224 | |||
| 225 | void BSD::GetPeerName(Kernel::HLERequestContext& ctx) { | ||
| 226 | IPC::RequestParser rp{ctx}; | ||
| 227 | const s32 fd = rp.Pop<s32>(); | ||
| 228 | |||
| 229 | LOG_DEBUG(Service, "called. fd={}", fd); | ||
| 69 | 230 | ||
| 231 | std::vector<u8> write_buffer(ctx.GetWriteBufferSize()); | ||
| 232 | const Errno bsd_errno = GetPeerNameImpl(fd, write_buffer); | ||
| 233 | |||
| 234 | ctx.WriteBuffer(write_buffer); | ||
| 235 | |||
| 236 | IPC::ResponseBuilder rb{ctx, 5}; | ||
| 70 | rb.Push(RESULT_SUCCESS); | 237 | rb.Push(RESULT_SUCCESS); |
| 71 | rb.Push<u32>(0); // ret | 238 | rb.Push<s32>(bsd_errno != Errno::SUCCESS ? -1 : 0); |
| 72 | rb.Push<u32>(0); // bsd errno | 239 | rb.PushEnum(bsd_errno); |
| 240 | rb.Push<u32>(static_cast<u32>(write_buffer.size())); | ||
| 241 | } | ||
| 242 | |||
| 243 | void BSD::GetSockName(Kernel::HLERequestContext& ctx) { | ||
| 244 | IPC::RequestParser rp{ctx}; | ||
| 245 | const s32 fd = rp.Pop<s32>(); | ||
| 246 | |||
| 247 | LOG_DEBUG(Service, "called. fd={}", fd); | ||
| 248 | |||
| 249 | std::vector<u8> write_buffer(ctx.GetWriteBufferSize()); | ||
| 250 | const Errno bsd_errno = GetSockNameImpl(fd, write_buffer); | ||
| 251 | |||
| 252 | ctx.WriteBuffer(write_buffer); | ||
| 253 | |||
| 254 | IPC::ResponseBuilder rb{ctx, 5}; | ||
| 255 | rb.Push(RESULT_SUCCESS); | ||
| 256 | rb.Push<s32>(bsd_errno != Errno::SUCCESS ? -1 : 0); | ||
| 257 | rb.PushEnum(bsd_errno); | ||
| 258 | rb.Push<u32>(static_cast<u32>(write_buffer.size())); | ||
| 73 | } | 259 | } |
| 74 | 260 | ||
| 75 | void BSD::Listen(Kernel::HLERequestContext& ctx) { | 261 | void BSD::Listen(Kernel::HLERequestContext& ctx) { |
| 76 | LOG_WARNING(Service, "(STUBBED) called"); | 262 | IPC::RequestParser rp{ctx}; |
| 263 | const s32 fd = rp.Pop<s32>(); | ||
| 264 | const s32 backlog = rp.Pop<s32>(); | ||
| 77 | 265 | ||
| 78 | IPC::ResponseBuilder rb{ctx, 4}; | 266 | LOG_DEBUG(Service, "called. fd={} backlog={}", fd, backlog); |
| 267 | |||
| 268 | BuildErrnoResponse(ctx, ListenImpl(fd, backlog)); | ||
| 269 | } | ||
| 270 | |||
| 271 | void BSD::Fcntl(Kernel::HLERequestContext& ctx) { | ||
| 272 | IPC::RequestParser rp{ctx}; | ||
| 273 | const s32 fd = rp.Pop<s32>(); | ||
| 274 | const s32 cmd = rp.Pop<s32>(); | ||
| 275 | const s32 arg = rp.Pop<s32>(); | ||
| 79 | 276 | ||
| 277 | LOG_DEBUG(Service, "called. fd={} cmd={} arg={}", fd, cmd, arg); | ||
| 278 | |||
| 279 | const auto [ret, bsd_errno] = FcntlImpl(fd, static_cast<FcntlCmd>(cmd), arg); | ||
| 280 | |||
| 281 | IPC::ResponseBuilder rb{ctx, 4}; | ||
| 80 | rb.Push(RESULT_SUCCESS); | 282 | rb.Push(RESULT_SUCCESS); |
| 81 | rb.Push<u32>(0); // ret | 283 | rb.Push<s32>(ret); |
| 82 | rb.Push<u32>(0); // bsd errno | 284 | rb.PushEnum(bsd_errno); |
| 83 | } | 285 | } |
| 84 | 286 | ||
| 85 | void BSD::SetSockOpt(Kernel::HLERequestContext& ctx) { | 287 | void BSD::SetSockOpt(Kernel::HLERequestContext& ctx) { |
| 86 | LOG_WARNING(Service, "(STUBBED) called"); | 288 | IPC::RequestParser rp{ctx}; |
| 87 | 289 | ||
| 88 | IPC::ResponseBuilder rb{ctx, 4}; | 290 | const s32 fd = rp.Pop<s32>(); |
| 291 | const u32 level = rp.Pop<u32>(); | ||
| 292 | const OptName optname = static_cast<OptName>(rp.Pop<u32>()); | ||
| 89 | 293 | ||
| 90 | rb.Push(RESULT_SUCCESS); | 294 | const std::vector<u8> buffer = ctx.ReadBuffer(); |
| 91 | rb.Push<u32>(0); // ret | 295 | const u8* optval = buffer.empty() ? nullptr : buffer.data(); |
| 92 | rb.Push<u32>(0); // bsd errno | 296 | size_t optlen = buffer.size(); |
| 297 | |||
| 298 | std::array<u64, 2> values; | ||
| 299 | if ((optname == OptName::SNDTIMEO || optname == OptName::RCVTIMEO) && buffer.size() == 8) { | ||
| 300 | std::memcpy(values.data(), buffer.data(), sizeof(values)); | ||
| 301 | optlen = sizeof(values); | ||
| 302 | optval = reinterpret_cast<const u8*>(values.data()); | ||
| 303 | } | ||
| 304 | |||
| 305 | LOG_DEBUG(Service, "called. fd={} level={} optname=0x{:x} optlen={}", fd, level, | ||
| 306 | static_cast<u32>(optname), optlen); | ||
| 307 | |||
| 308 | BuildErrnoResponse(ctx, SetSockOptImpl(fd, level, optname, optlen, optval)); | ||
| 309 | } | ||
| 310 | |||
| 311 | void BSD::Shutdown(Kernel::HLERequestContext& ctx) { | ||
| 312 | IPC::RequestParser rp{ctx}; | ||
| 313 | |||
| 314 | const s32 fd = rp.Pop<s32>(); | ||
| 315 | const s32 how = rp.Pop<s32>(); | ||
| 316 | |||
| 317 | LOG_DEBUG(Service, "called. fd={} how={}", fd, how); | ||
| 318 | |||
| 319 | BuildErrnoResponse(ctx, ShutdownImpl(fd, how)); | ||
| 320 | } | ||
| 321 | |||
| 322 | void BSD::Recv(Kernel::HLERequestContext& ctx) { | ||
| 323 | IPC::RequestParser rp{ctx}; | ||
| 324 | |||
| 325 | const s32 fd = rp.Pop<s32>(); | ||
| 326 | const u32 flags = rp.Pop<u32>(); | ||
| 327 | |||
| 328 | LOG_DEBUG(Service, "called. fd={} flags=0x{:x} len={}", fd, flags, ctx.GetWriteBufferSize()); | ||
| 329 | |||
| 330 | ExecuteWork(ctx, "BSD:Recv", IsBlockingSocket(fd), | ||
| 331 | RecvWork{ | ||
| 332 | .fd = fd, | ||
| 333 | .flags = flags, | ||
| 334 | .message = std::vector<u8>(ctx.GetWriteBufferSize()), | ||
| 335 | }); | ||
| 336 | } | ||
| 337 | |||
| 338 | void BSD::RecvFrom(Kernel::HLERequestContext& ctx) { | ||
| 339 | IPC::RequestParser rp{ctx}; | ||
| 340 | |||
| 341 | const s32 fd = rp.Pop<s32>(); | ||
| 342 | const u32 flags = rp.Pop<u32>(); | ||
| 343 | |||
| 344 | LOG_DEBUG(Service, "called. fd={} flags=0x{:x} len={} addrlen={}", fd, flags, | ||
| 345 | ctx.GetWriteBufferSize(0), ctx.GetWriteBufferSize(1)); | ||
| 346 | |||
| 347 | ExecuteWork(ctx, "BSD:RecvFrom", IsBlockingSocket(fd), | ||
| 348 | RecvFromWork{ | ||
| 349 | .fd = fd, | ||
| 350 | .flags = flags, | ||
| 351 | .message = std::vector<u8>(ctx.GetWriteBufferSize(0)), | ||
| 352 | .addr = std::vector<u8>(ctx.GetWriteBufferSize(1)), | ||
| 353 | }); | ||
| 354 | } | ||
| 355 | |||
| 356 | void BSD::Send(Kernel::HLERequestContext& ctx) { | ||
| 357 | IPC::RequestParser rp{ctx}; | ||
| 358 | |||
| 359 | const s32 fd = rp.Pop<s32>(); | ||
| 360 | const u32 flags = rp.Pop<u32>(); | ||
| 361 | |||
| 362 | LOG_DEBUG(Service, "called. fd={} flags=0x{:x} len={}", fd, flags, ctx.GetReadBufferSize()); | ||
| 363 | |||
| 364 | ExecuteWork(ctx, "BSD:Send", IsBlockingSocket(fd), | ||
| 365 | SendWork{ | ||
| 366 | .fd = fd, | ||
| 367 | .flags = flags, | ||
| 368 | .message = ctx.ReadBuffer(), | ||
| 369 | }); | ||
| 93 | } | 370 | } |
| 94 | 371 | ||
| 95 | void BSD::SendTo(Kernel::HLERequestContext& ctx) { | 372 | void BSD::SendTo(Kernel::HLERequestContext& ctx) { |
| 96 | LOG_WARNING(Service, "(STUBBED) called"); | 373 | IPC::RequestParser rp{ctx}; |
| 374 | const s32 fd = rp.Pop<s32>(); | ||
| 375 | const u32 flags = rp.Pop<u32>(); | ||
| 376 | |||
| 377 | LOG_DEBUG(Service, "called. fd={} flags=0x{} len={} addrlen={}", fd, flags, | ||
| 378 | ctx.GetReadBufferSize(0), ctx.GetReadBufferSize(1)); | ||
| 379 | |||
| 380 | ExecuteWork(ctx, "BSD:SendTo", IsBlockingSocket(fd), | ||
| 381 | SendToWork{ | ||
| 382 | .fd = fd, | ||
| 383 | .flags = flags, | ||
| 384 | .message = ctx.ReadBuffer(0), | ||
| 385 | .addr = ctx.ReadBuffer(1), | ||
| 386 | }); | ||
| 387 | } | ||
| 97 | 388 | ||
| 98 | IPC::ResponseBuilder rb{ctx, 4}; | 389 | void BSD::Write(Kernel::HLERequestContext& ctx) { |
| 390 | IPC::RequestParser rp{ctx}; | ||
| 391 | const s32 fd = rp.Pop<s32>(); | ||
| 99 | 392 | ||
| 100 | rb.Push(RESULT_SUCCESS); | 393 | LOG_DEBUG(Service, "called. fd={} len={}", fd, ctx.GetReadBufferSize()); |
| 101 | rb.Push<u32>(0); // ret | 394 | |
| 102 | rb.Push<u32>(0); // bsd errno | 395 | ExecuteWork(ctx, "BSD:Write", IsBlockingSocket(fd), |
| 396 | SendWork{ | ||
| 397 | .fd = fd, | ||
| 398 | .flags = 0, | ||
| 399 | .message = ctx.ReadBuffer(), | ||
| 400 | }); | ||
| 103 | } | 401 | } |
| 104 | 402 | ||
| 105 | void BSD::Close(Kernel::HLERequestContext& ctx) { | 403 | void BSD::Close(Kernel::HLERequestContext& ctx) { |
| 106 | LOG_WARNING(Service, "(STUBBED) called"); | 404 | IPC::RequestParser rp{ctx}; |
| 405 | const s32 fd = rp.Pop<s32>(); | ||
| 406 | |||
| 407 | LOG_DEBUG(Service, "called. fd={}", fd); | ||
| 408 | |||
| 409 | BuildErrnoResponse(ctx, CloseImpl(fd)); | ||
| 410 | } | ||
| 411 | |||
| 412 | template <typename Work> | ||
| 413 | void BSD::ExecuteWork(Kernel::HLERequestContext& ctx, std::string_view sleep_reason, | ||
| 414 | bool is_blocking, Work work) { | ||
| 415 | if (!is_blocking) { | ||
| 416 | work.Execute(this); | ||
| 417 | work.Response(ctx); | ||
| 418 | return; | ||
| 419 | } | ||
| 420 | |||
| 421 | // Signal a dummy response to make IPC validation happy | ||
| 422 | // This will be overwritten by the SleepClientThread callback | ||
| 423 | work.Response(ctx); | ||
| 424 | |||
| 425 | auto worker = worker_pool.CaptureWorker(); | ||
| 426 | |||
| 427 | ctx.SleepClientThread(std::string(sleep_reason), std::numeric_limits<u64>::max(), | ||
| 428 | worker->Callback<Work>(), worker->KernelEvent()); | ||
| 429 | |||
| 430 | worker->SendWork(std::move(work)); | ||
| 431 | } | ||
| 432 | |||
| 433 | std::pair<s32, Errno> BSD::SocketImpl(Domain domain, Type type, Protocol protocol) { | ||
| 434 | if (type == Type::SEQPACKET) { | ||
| 435 | UNIMPLEMENTED_MSG("SOCK_SEQPACKET errno management"); | ||
| 436 | } else if (type == Type::RAW && (domain != Domain::INET || protocol != Protocol::ICMP)) { | ||
| 437 | UNIMPLEMENTED_MSG("SOCK_RAW errno management"); | ||
| 438 | } | ||
| 439 | |||
| 440 | [[maybe_unused]] const bool unk_flag = (static_cast<u32>(type) & 0x20000000) != 0; | ||
| 441 | UNIMPLEMENTED_IF_MSG(unk_flag, "Unknown flag in type"); | ||
| 442 | type = static_cast<Type>(static_cast<u32>(type) & ~0x20000000); | ||
| 443 | |||
| 444 | const s32 fd = FindFreeFileDescriptorHandle(); | ||
| 445 | if (fd < 0) { | ||
| 446 | LOG_ERROR(Service, "No more file descriptors available"); | ||
| 447 | return {-1, Errno::MFILE}; | ||
| 448 | } | ||
| 449 | |||
| 450 | FileDescriptor& descriptor = file_descriptors[fd].emplace(); | ||
| 451 | // ENONMEM might be thrown here | ||
| 452 | |||
| 453 | LOG_INFO(Service, "New socket fd={}", fd); | ||
| 454 | |||
| 455 | descriptor.socket = std::make_unique<Network::Socket>(); | ||
| 456 | descriptor.socket->Initialize(Translate(domain), Translate(type), Translate(type, protocol)); | ||
| 457 | descriptor.is_connection_based = IsConnectionBased(type); | ||
| 458 | |||
| 459 | return {fd, Errno::SUCCESS}; | ||
| 460 | } | ||
| 107 | 461 | ||
| 462 | std::pair<s32, Errno> BSD::PollImpl(std::vector<u8>& write_buffer, std::vector<u8> read_buffer, | ||
| 463 | s32 nfds, s32 timeout) { | ||
| 464 | if (write_buffer.size() < nfds * sizeof(PollFD)) { | ||
| 465 | return {-1, Errno::INVAL}; | ||
| 466 | } | ||
| 467 | |||
| 468 | if (nfds == 0) { | ||
| 469 | // When no entries are provided, -1 is returned with errno zero | ||
| 470 | return {-1, Errno::SUCCESS}; | ||
| 471 | } | ||
| 472 | |||
| 473 | const size_t length = std::min(read_buffer.size(), write_buffer.size()); | ||
| 474 | std::vector<PollFD> fds(nfds); | ||
| 475 | std::memcpy(fds.data(), read_buffer.data(), length); | ||
| 476 | |||
| 477 | if (timeout >= 0) { | ||
| 478 | const s64 seconds = timeout / 1000; | ||
| 479 | const u64 nanoseconds = 1'000'000 * (static_cast<u64>(timeout) % 1000); | ||
| 480 | |||
| 481 | if (seconds < 0) { | ||
| 482 | return {-1, Errno::INVAL}; | ||
| 483 | } | ||
| 484 | if (nanoseconds > 999'999'999) { | ||
| 485 | return {-1, Errno::INVAL}; | ||
| 486 | } | ||
| 487 | } else if (timeout != -1) { | ||
| 488 | return {-1, Errno::INVAL}; | ||
| 489 | } | ||
| 490 | |||
| 491 | for (PollFD& pollfd : fds) { | ||
| 492 | ASSERT(pollfd.revents == 0); | ||
| 493 | |||
| 494 | if (pollfd.fd > MAX_FD || pollfd.fd < 0) { | ||
| 495 | LOG_ERROR(Service, "File descriptor handle={} is invalid", pollfd.fd); | ||
| 496 | pollfd.revents = 0; | ||
| 497 | return {0, Errno::SUCCESS}; | ||
| 498 | } | ||
| 499 | |||
| 500 | std::optional<FileDescriptor>& descriptor = file_descriptors[pollfd.fd]; | ||
| 501 | if (!descriptor) { | ||
| 502 | LOG_ERROR(Service, "File descriptor handle={} is not allocated", pollfd.fd); | ||
| 503 | pollfd.revents = POLL_NVAL; | ||
| 504 | return {0, Errno::SUCCESS}; | ||
| 505 | } | ||
| 506 | } | ||
| 507 | |||
| 508 | std::vector<Network::PollFD> host_pollfds(fds.size()); | ||
| 509 | std::transform(fds.begin(), fds.end(), host_pollfds.begin(), [this](PollFD pollfd) { | ||
| 510 | Network::PollFD result; | ||
| 511 | result.socket = file_descriptors[pollfd.fd]->socket.get(); | ||
| 512 | result.events = TranslatePollEventsToHost(pollfd.events); | ||
| 513 | result.revents = 0; | ||
| 514 | return result; | ||
| 515 | }); | ||
| 516 | |||
| 517 | const auto result = Network::Poll(host_pollfds, timeout); | ||
| 518 | |||
| 519 | const size_t num = host_pollfds.size(); | ||
| 520 | for (size_t i = 0; i < num; ++i) { | ||
| 521 | fds[i].revents = TranslatePollEventsToGuest(host_pollfds[i].revents); | ||
| 522 | } | ||
| 523 | std::memcpy(write_buffer.data(), fds.data(), length); | ||
| 524 | |||
| 525 | return Translate(result); | ||
| 526 | } | ||
| 527 | |||
| 528 | std::pair<s32, Errno> BSD::AcceptImpl(s32 fd, std::vector<u8>& write_buffer) { | ||
| 529 | if (!IsFileDescriptorValid(fd)) { | ||
| 530 | return {-1, Errno::BADF}; | ||
| 531 | } | ||
| 532 | |||
| 533 | const s32 new_fd = FindFreeFileDescriptorHandle(); | ||
| 534 | if (new_fd < 0) { | ||
| 535 | LOG_ERROR(Service, "No more file descriptors available"); | ||
| 536 | return {-1, Errno::MFILE}; | ||
| 537 | } | ||
| 538 | |||
| 539 | FileDescriptor& descriptor = *file_descriptors[fd]; | ||
| 540 | auto [result, bsd_errno] = descriptor.socket->Accept(); | ||
| 541 | if (bsd_errno != Network::Errno::SUCCESS) { | ||
| 542 | return {-1, Translate(bsd_errno)}; | ||
| 543 | } | ||
| 544 | |||
| 545 | FileDescriptor& new_descriptor = file_descriptors[new_fd].emplace(); | ||
| 546 | new_descriptor.socket = std::move(result.socket); | ||
| 547 | new_descriptor.is_connection_based = descriptor.is_connection_based; | ||
| 548 | |||
| 549 | ASSERT(write_buffer.size() == sizeof(SockAddrIn)); | ||
| 550 | const SockAddrIn guest_addr_in = Translate(result.sockaddr_in); | ||
| 551 | std::memcpy(write_buffer.data(), &guest_addr_in, sizeof(guest_addr_in)); | ||
| 552 | |||
| 553 | return {new_fd, Errno::SUCCESS}; | ||
| 554 | } | ||
| 555 | |||
| 556 | Errno BSD::BindImpl(s32 fd, const std::vector<u8>& addr) { | ||
| 557 | if (!IsFileDescriptorValid(fd)) { | ||
| 558 | return Errno::BADF; | ||
| 559 | } | ||
| 560 | ASSERT(addr.size() == sizeof(SockAddrIn)); | ||
| 561 | SockAddrIn addr_in; | ||
| 562 | std::memcpy(&addr_in, addr.data(), sizeof(addr_in)); | ||
| 563 | |||
| 564 | return Translate(file_descriptors[fd]->socket->Bind(Translate(addr_in))); | ||
| 565 | } | ||
| 566 | |||
| 567 | Errno BSD::ConnectImpl(s32 fd, const std::vector<u8>& addr) { | ||
| 568 | if (!IsFileDescriptorValid(fd)) { | ||
| 569 | return Errno::BADF; | ||
| 570 | } | ||
| 571 | |||
| 572 | UNIMPLEMENTED_IF(addr.size() != sizeof(SockAddrIn)); | ||
| 573 | SockAddrIn addr_in; | ||
| 574 | std::memcpy(&addr_in, addr.data(), sizeof(addr_in)); | ||
| 575 | |||
| 576 | return Translate(file_descriptors[fd]->socket->Connect(Translate(addr_in))); | ||
| 577 | } | ||
| 578 | |||
| 579 | Errno BSD::GetPeerNameImpl(s32 fd, std::vector<u8>& write_buffer) { | ||
| 580 | if (!IsFileDescriptorValid(fd)) { | ||
| 581 | return Errno::BADF; | ||
| 582 | } | ||
| 583 | |||
| 584 | const auto [addr_in, bsd_errno] = file_descriptors[fd]->socket->GetPeerName(); | ||
| 585 | if (bsd_errno != Network::Errno::SUCCESS) { | ||
| 586 | return Translate(bsd_errno); | ||
| 587 | } | ||
| 588 | const SockAddrIn guest_addrin = Translate(addr_in); | ||
| 589 | |||
| 590 | ASSERT(write_buffer.size() == sizeof(guest_addrin)); | ||
| 591 | std::memcpy(write_buffer.data(), &guest_addrin, sizeof(guest_addrin)); | ||
| 592 | return Translate(bsd_errno); | ||
| 593 | } | ||
| 594 | |||
| 595 | Errno BSD::GetSockNameImpl(s32 fd, std::vector<u8>& write_buffer) { | ||
| 596 | if (!IsFileDescriptorValid(fd)) { | ||
| 597 | return Errno::BADF; | ||
| 598 | } | ||
| 599 | |||
| 600 | const auto [addr_in, bsd_errno] = file_descriptors[fd]->socket->GetSockName(); | ||
| 601 | if (bsd_errno != Network::Errno::SUCCESS) { | ||
| 602 | return Translate(bsd_errno); | ||
| 603 | } | ||
| 604 | const SockAddrIn guest_addrin = Translate(addr_in); | ||
| 605 | |||
| 606 | ASSERT(write_buffer.size() == sizeof(guest_addrin)); | ||
| 607 | std::memcpy(write_buffer.data(), &guest_addrin, sizeof(guest_addrin)); | ||
| 608 | return Translate(bsd_errno); | ||
| 609 | } | ||
| 610 | |||
| 611 | Errno BSD::ListenImpl(s32 fd, s32 backlog) { | ||
| 612 | if (!IsFileDescriptorValid(fd)) { | ||
| 613 | return Errno::BADF; | ||
| 614 | } | ||
| 615 | return Translate(file_descriptors[fd]->socket->Listen(backlog)); | ||
| 616 | } | ||
| 617 | |||
| 618 | std::pair<s32, Errno> BSD::FcntlImpl(s32 fd, FcntlCmd cmd, s32 arg) { | ||
| 619 | if (!IsFileDescriptorValid(fd)) { | ||
| 620 | return {-1, Errno::BADF}; | ||
| 621 | } | ||
| 622 | |||
| 623 | FileDescriptor& descriptor = *file_descriptors[fd]; | ||
| 624 | |||
| 625 | switch (cmd) { | ||
| 626 | case FcntlCmd::GETFL: | ||
| 627 | ASSERT(arg == 0); | ||
| 628 | return {descriptor.flags, Errno::SUCCESS}; | ||
| 629 | case FcntlCmd::SETFL: { | ||
| 630 | const bool enable = (arg & FLAG_O_NONBLOCK) != 0; | ||
| 631 | const Errno bsd_errno = Translate(descriptor.socket->SetNonBlock(enable)); | ||
| 632 | if (bsd_errno != Errno::SUCCESS) { | ||
| 633 | return {-1, bsd_errno}; | ||
| 634 | } | ||
| 635 | descriptor.flags = arg; | ||
| 636 | return {0, Errno::SUCCESS}; | ||
| 637 | } | ||
| 638 | default: | ||
| 639 | UNIMPLEMENTED_MSG("Unimplemented cmd={}", static_cast<int>(cmd)); | ||
| 640 | return {-1, Errno::SUCCESS}; | ||
| 641 | } | ||
| 642 | } | ||
| 643 | |||
| 644 | Errno BSD::SetSockOptImpl(s32 fd, u32 level, OptName optname, size_t optlen, const void* optval) { | ||
| 645 | UNIMPLEMENTED_IF(level != 0xffff); // SOL_SOCKET | ||
| 646 | |||
| 647 | if (!IsFileDescriptorValid(fd)) { | ||
| 648 | return Errno::BADF; | ||
| 649 | } | ||
| 650 | |||
| 651 | Network::Socket* const socket = file_descriptors[fd]->socket.get(); | ||
| 652 | |||
| 653 | if (optname == OptName::LINGER) { | ||
| 654 | ASSERT(optlen == sizeof(Linger)); | ||
| 655 | Linger linger; | ||
| 656 | std::memcpy(&linger, optval, sizeof(linger)); | ||
| 657 | ASSERT(linger.onoff == 0 || linger.onoff == 1); | ||
| 658 | |||
| 659 | return Translate(socket->SetLinger(linger.onoff != 0, linger.linger)); | ||
| 660 | } | ||
| 661 | |||
| 662 | ASSERT(optlen == sizeof(u32)); | ||
| 663 | u32 value; | ||
| 664 | std::memcpy(&value, optval, sizeof(value)); | ||
| 665 | |||
| 666 | switch (optname) { | ||
| 667 | case OptName::REUSEADDR: | ||
| 668 | ASSERT(value == 0 || value == 1); | ||
| 669 | return Translate(socket->SetReuseAddr(value != 0)); | ||
| 670 | case OptName::BROADCAST: | ||
| 671 | ASSERT(value == 0 || value == 1); | ||
| 672 | return Translate(socket->SetBroadcast(value != 0)); | ||
| 673 | case OptName::SNDBUF: | ||
| 674 | return Translate(socket->SetSndBuf(value)); | ||
| 675 | case OptName::RCVBUF: | ||
| 676 | return Translate(socket->SetRcvBuf(value)); | ||
| 677 | case OptName::SNDTIMEO: | ||
| 678 | return Translate(socket->SetSndTimeo(value)); | ||
| 679 | case OptName::RCVTIMEO: | ||
| 680 | return Translate(socket->SetRcvTimeo(value)); | ||
| 681 | default: | ||
| 682 | UNIMPLEMENTED_MSG("Unimplemented optname={}", static_cast<int>(optname)); | ||
| 683 | return Errno::SUCCESS; | ||
| 684 | } | ||
| 685 | } | ||
| 686 | |||
| 687 | Errno BSD::ShutdownImpl(s32 fd, s32 how) { | ||
| 688 | if (!IsFileDescriptorValid(fd)) { | ||
| 689 | return Errno::BADF; | ||
| 690 | } | ||
| 691 | const Network::ShutdownHow host_how = Translate(static_cast<ShutdownHow>(how)); | ||
| 692 | return Translate(file_descriptors[fd]->socket->Shutdown(host_how)); | ||
| 693 | } | ||
| 694 | |||
| 695 | std::pair<s32, Errno> BSD::RecvImpl(s32 fd, u32 flags, std::vector<u8>& message) { | ||
| 696 | if (!IsFileDescriptorValid(fd)) { | ||
| 697 | return {-1, Errno::BADF}; | ||
| 698 | } | ||
| 699 | return Translate(file_descriptors[fd]->socket->Recv(flags, message)); | ||
| 700 | } | ||
| 701 | |||
| 702 | std::pair<s32, Errno> BSD::RecvFromImpl(s32 fd, u32 flags, std::vector<u8>& message, | ||
| 703 | std::vector<u8>& addr) { | ||
| 704 | if (!IsFileDescriptorValid(fd)) { | ||
| 705 | return {-1, Errno::BADF}; | ||
| 706 | } | ||
| 707 | |||
| 708 | FileDescriptor& descriptor = *file_descriptors[fd]; | ||
| 709 | |||
| 710 | Network::SockAddrIn addr_in{}; | ||
| 711 | Network::SockAddrIn* p_addr_in = nullptr; | ||
| 712 | if (descriptor.is_connection_based) { | ||
| 713 | // Connection based file descriptors (e.g. TCP) zero addr | ||
| 714 | addr.clear(); | ||
| 715 | } else { | ||
| 716 | p_addr_in = &addr_in; | ||
| 717 | } | ||
| 718 | |||
| 719 | // Apply flags | ||
| 720 | if ((flags & FLAG_MSG_DONTWAIT) != 0) { | ||
| 721 | flags &= ~FLAG_MSG_DONTWAIT; | ||
| 722 | if ((descriptor.flags & FLAG_O_NONBLOCK) == 0) { | ||
| 723 | descriptor.socket->SetNonBlock(true); | ||
| 724 | } | ||
| 725 | } | ||
| 726 | |||
| 727 | const auto [ret, bsd_errno] = Translate(descriptor.socket->RecvFrom(flags, message, p_addr_in)); | ||
| 728 | |||
| 729 | // Restore original state | ||
| 730 | if ((descriptor.flags & FLAG_O_NONBLOCK) == 0) { | ||
| 731 | descriptor.socket->SetNonBlock(false); | ||
| 732 | } | ||
| 733 | |||
| 734 | if (p_addr_in) { | ||
| 735 | if (ret < 0) { | ||
| 736 | addr.clear(); | ||
| 737 | } else { | ||
| 738 | ASSERT(addr.size() == sizeof(SockAddrIn)); | ||
| 739 | const SockAddrIn result = Translate(addr_in); | ||
| 740 | std::memcpy(addr.data(), &result, sizeof(result)); | ||
| 741 | } | ||
| 742 | } | ||
| 743 | |||
| 744 | return {ret, bsd_errno}; | ||
| 745 | } | ||
| 746 | |||
| 747 | std::pair<s32, Errno> BSD::SendImpl(s32 fd, u32 flags, const std::vector<u8>& message) { | ||
| 748 | if (!IsFileDescriptorValid(fd)) { | ||
| 749 | return {-1, Errno::BADF}; | ||
| 750 | } | ||
| 751 | return Translate(file_descriptors[fd]->socket->Send(message, flags)); | ||
| 752 | } | ||
| 753 | |||
| 754 | std::pair<s32, Errno> BSD::SendToImpl(s32 fd, u32 flags, const std::vector<u8>& message, | ||
| 755 | const std::vector<u8>& addr) { | ||
| 756 | if (!IsFileDescriptorValid(fd)) { | ||
| 757 | return {-1, Errno::BADF}; | ||
| 758 | } | ||
| 759 | |||
| 760 | Network::SockAddrIn addr_in; | ||
| 761 | Network::SockAddrIn* p_addr_in = nullptr; | ||
| 762 | if (!addr.empty()) { | ||
| 763 | ASSERT(addr.size() == sizeof(SockAddrIn)); | ||
| 764 | SockAddrIn guest_addr_in; | ||
| 765 | std::memcpy(&guest_addr_in, addr.data(), sizeof(guest_addr_in)); | ||
| 766 | addr_in = Translate(guest_addr_in); | ||
| 767 | } | ||
| 768 | |||
| 769 | return Translate(file_descriptors[fd]->socket->SendTo(flags, message, p_addr_in)); | ||
| 770 | } | ||
| 771 | |||
| 772 | Errno BSD::CloseImpl(s32 fd) { | ||
| 773 | if (!IsFileDescriptorValid(fd)) { | ||
| 774 | return Errno::BADF; | ||
| 775 | } | ||
| 776 | |||
| 777 | const Errno bsd_errno = Translate(file_descriptors[fd]->socket->Close()); | ||
| 778 | if (bsd_errno != Errno::SUCCESS) { | ||
| 779 | return bsd_errno; | ||
| 780 | } | ||
| 781 | |||
| 782 | LOG_INFO(Service, "Close socket fd={}", fd); | ||
| 783 | |||
| 784 | file_descriptors[fd].reset(); | ||
| 785 | return bsd_errno; | ||
| 786 | } | ||
| 787 | |||
| 788 | s32 BSD::FindFreeFileDescriptorHandle() noexcept { | ||
| 789 | for (s32 fd = 0; fd < static_cast<s32>(file_descriptors.size()); ++fd) { | ||
| 790 | if (!file_descriptors[fd]) { | ||
| 791 | return fd; | ||
| 792 | } | ||
| 793 | } | ||
| 794 | return -1; | ||
| 795 | } | ||
| 796 | |||
| 797 | bool BSD::IsFileDescriptorValid(s32 fd) const noexcept { | ||
| 798 | if (fd > MAX_FD || fd < 0) { | ||
| 799 | LOG_ERROR(Service, "Invalid file descriptor handle={}", fd); | ||
| 800 | return false; | ||
| 801 | } | ||
| 802 | if (!file_descriptors[fd]) { | ||
| 803 | LOG_ERROR(Service, "File descriptor handle={} is not allocated", fd); | ||
| 804 | return false; | ||
| 805 | } | ||
| 806 | return true; | ||
| 807 | } | ||
| 808 | |||
| 809 | bool BSD::IsBlockingSocket(s32 fd) const noexcept { | ||
| 810 | // Inform invalid sockets as non-blocking | ||
| 811 | // This way we avoid using a worker thread as it will fail without blocking host | ||
| 812 | if (fd > MAX_FD || fd < 0) { | ||
| 813 | return false; | ||
| 814 | } | ||
| 815 | if (!file_descriptors[fd]) { | ||
| 816 | return false; | ||
| 817 | } | ||
| 818 | return (file_descriptors[fd]->flags & FLAG_O_NONBLOCK) != 0; | ||
| 819 | } | ||
| 820 | |||
| 821 | void BSD::BuildErrnoResponse(Kernel::HLERequestContext& ctx, Errno bsd_errno) const noexcept { | ||
| 108 | IPC::ResponseBuilder rb{ctx, 4}; | 822 | IPC::ResponseBuilder rb{ctx, 4}; |
| 109 | 823 | ||
| 110 | rb.Push(RESULT_SUCCESS); | 824 | rb.Push(RESULT_SUCCESS); |
| 111 | rb.Push<u32>(0); // ret | 825 | rb.Push<s32>(bsd_errno == Errno::SUCCESS ? 0 : -1); |
| 112 | rb.Push<u32>(0); // bsd errno | 826 | rb.PushEnum(bsd_errno); |
| 113 | } | 827 | } |
| 114 | 828 | ||
| 115 | BSD::BSD(const char* name) : ServiceFramework(name) { | 829 | BSD::BSD(Core::System& system, const char* name) |
| 830 | : ServiceFramework(name), worker_pool{system, this} { | ||
| 116 | // clang-format off | 831 | // clang-format off |
| 117 | static const FunctionInfo functions[] = { | 832 | static const FunctionInfo functions[] = { |
| 118 | {0, &BSD::RegisterClient, "RegisterClient"}, | 833 | {0, &BSD::RegisterClient, "RegisterClient"}, |
| @@ -121,25 +836,25 @@ BSD::BSD(const char* name) : ServiceFramework(name) { | |||
| 121 | {3, nullptr, "SocketExempt"}, | 836 | {3, nullptr, "SocketExempt"}, |
| 122 | {4, nullptr, "Open"}, | 837 | {4, nullptr, "Open"}, |
| 123 | {5, &BSD::Select, "Select"}, | 838 | {5, &BSD::Select, "Select"}, |
| 124 | {6, nullptr, "Poll"}, | 839 | {6, &BSD::Poll, "Poll"}, |
| 125 | {7, nullptr, "Sysctl"}, | 840 | {7, nullptr, "Sysctl"}, |
| 126 | {8, nullptr, "Recv"}, | 841 | {8, &BSD::Recv, "Recv"}, |
| 127 | {9, nullptr, "RecvFrom"}, | 842 | {9, &BSD::RecvFrom, "RecvFrom"}, |
| 128 | {10, nullptr, "Send"}, | 843 | {10, &BSD::Send, "Send"}, |
| 129 | {11, &BSD::SendTo, "SendTo"}, | 844 | {11, &BSD::SendTo, "SendTo"}, |
| 130 | {12, nullptr, "Accept"}, | 845 | {12, &BSD::Accept, "Accept"}, |
| 131 | {13, &BSD::Bind, "Bind"}, | 846 | {13, &BSD::Bind, "Bind"}, |
| 132 | {14, &BSD::Connect, "Connect"}, | 847 | {14, &BSD::Connect, "Connect"}, |
| 133 | {15, nullptr, "GetPeerName"}, | 848 | {15, &BSD::GetPeerName, "GetPeerName"}, |
| 134 | {16, nullptr, "GetSockName"}, | 849 | {16, &BSD::GetSockName, "GetSockName"}, |
| 135 | {17, nullptr, "GetSockOpt"}, | 850 | {17, nullptr, "GetSockOpt"}, |
| 136 | {18, &BSD::Listen, "Listen"}, | 851 | {18, &BSD::Listen, "Listen"}, |
| 137 | {19, nullptr, "Ioctl"}, | 852 | {19, nullptr, "Ioctl"}, |
| 138 | {20, nullptr, "Fcntl"}, | 853 | {20, &BSD::Fcntl, "Fcntl"}, |
| 139 | {21, &BSD::SetSockOpt, "SetSockOpt"}, | 854 | {21, &BSD::SetSockOpt, "SetSockOpt"}, |
| 140 | {22, nullptr, "Shutdown"}, | 855 | {22, &BSD::Shutdown, "Shutdown"}, |
| 141 | {23, nullptr, "ShutdownAllSockets"}, | 856 | {23, nullptr, "ShutdownAllSockets"}, |
| 142 | {24, nullptr, "Write"}, | 857 | {24, &BSD::Write, "Write"}, |
| 143 | {25, nullptr, "Read"}, | 858 | {25, nullptr, "Read"}, |
| 144 | {26, &BSD::Close, "Close"}, | 859 | {26, &BSD::Close, "Close"}, |
| 145 | {27, nullptr, "DuplicateSocket"}, | 860 | {27, nullptr, "DuplicateSocket"}, |
diff --git a/src/core/hle/service/sockets/bsd.h b/src/core/hle/service/sockets/bsd.h index 3098e3baf..357531951 100644 --- a/src/core/hle/service/sockets/bsd.h +++ b/src/core/hle/service/sockets/bsd.h | |||
| @@ -4,30 +4,174 @@ | |||
| 4 | 4 | ||
| 5 | #pragma once | 5 | #pragma once |
| 6 | 6 | ||
| 7 | #include <memory> | ||
| 8 | #include <string_view> | ||
| 9 | #include <vector> | ||
| 10 | |||
| 11 | #include "common/common_types.h" | ||
| 7 | #include "core/hle/kernel/hle_ipc.h" | 12 | #include "core/hle/kernel/hle_ipc.h" |
| 8 | #include "core/hle/service/service.h" | 13 | #include "core/hle/service/service.h" |
| 14 | #include "core/hle/service/sockets/blocking_worker.h" | ||
| 15 | #include "core/hle/service/sockets/sockets.h" | ||
| 16 | |||
| 17 | namespace Core { | ||
| 18 | class System; | ||
| 19 | } | ||
| 20 | |||
| 21 | namespace Network { | ||
| 22 | class Socket; | ||
| 23 | } | ||
| 9 | 24 | ||
| 10 | namespace Service::Sockets { | 25 | namespace Service::Sockets { |
| 11 | 26 | ||
| 12 | class BSD final : public ServiceFramework<BSD> { | 27 | class BSD final : public ServiceFramework<BSD> { |
| 13 | public: | 28 | public: |
| 14 | explicit BSD(const char* name); | 29 | explicit BSD(Core::System& system, const char* name); |
| 15 | ~BSD() override; | 30 | ~BSD() override; |
| 16 | 31 | ||
| 17 | private: | 32 | private: |
| 33 | /// Maximum number of file descriptors | ||
| 34 | static constexpr size_t MAX_FD = 128; | ||
| 35 | |||
| 36 | struct FileDescriptor { | ||
| 37 | std::unique_ptr<Network::Socket> socket; | ||
| 38 | s32 flags = 0; | ||
| 39 | bool is_connection_based = false; | ||
| 40 | }; | ||
| 41 | |||
| 42 | struct PollWork { | ||
| 43 | void Execute(BSD* bsd); | ||
| 44 | void Response(Kernel::HLERequestContext& ctx); | ||
| 45 | |||
| 46 | s32 nfds; | ||
| 47 | s32 timeout; | ||
| 48 | std::vector<u8> read_buffer; | ||
| 49 | std::vector<u8> write_buffer; | ||
| 50 | s32 ret{}; | ||
| 51 | Errno bsd_errno{}; | ||
| 52 | }; | ||
| 53 | |||
| 54 | struct AcceptWork { | ||
| 55 | void Execute(BSD* bsd); | ||
| 56 | void Response(Kernel::HLERequestContext& ctx); | ||
| 57 | |||
| 58 | s32 fd; | ||
| 59 | std::vector<u8> write_buffer; | ||
| 60 | s32 ret{}; | ||
| 61 | Errno bsd_errno{}; | ||
| 62 | }; | ||
| 63 | |||
| 64 | struct ConnectWork { | ||
| 65 | void Execute(BSD* bsd); | ||
| 66 | void Response(Kernel::HLERequestContext& ctx); | ||
| 67 | |||
| 68 | s32 fd; | ||
| 69 | std::vector<u8> addr; | ||
| 70 | Errno bsd_errno{}; | ||
| 71 | }; | ||
| 72 | |||
| 73 | struct RecvWork { | ||
| 74 | void Execute(BSD* bsd); | ||
| 75 | void Response(Kernel::HLERequestContext& ctx); | ||
| 76 | |||
| 77 | s32 fd; | ||
| 78 | u32 flags; | ||
| 79 | std::vector<u8> message; | ||
| 80 | s32 ret{}; | ||
| 81 | Errno bsd_errno{}; | ||
| 82 | }; | ||
| 83 | |||
| 84 | struct RecvFromWork { | ||
| 85 | void Execute(BSD* bsd); | ||
| 86 | void Response(Kernel::HLERequestContext& ctx); | ||
| 87 | |||
| 88 | s32 fd; | ||
| 89 | u32 flags; | ||
| 90 | std::vector<u8> message; | ||
| 91 | std::vector<u8> addr; | ||
| 92 | s32 ret{}; | ||
| 93 | Errno bsd_errno{}; | ||
| 94 | }; | ||
| 95 | |||
| 96 | struct SendWork { | ||
| 97 | void Execute(BSD* bsd); | ||
| 98 | void Response(Kernel::HLERequestContext& ctx); | ||
| 99 | |||
| 100 | s32 fd; | ||
| 101 | u32 flags; | ||
| 102 | std::vector<u8> message; | ||
| 103 | s32 ret{}; | ||
| 104 | Errno bsd_errno{}; | ||
| 105 | }; | ||
| 106 | |||
| 107 | struct SendToWork { | ||
| 108 | void Execute(BSD* bsd); | ||
| 109 | void Response(Kernel::HLERequestContext& ctx); | ||
| 110 | |||
| 111 | s32 fd; | ||
| 112 | u32 flags; | ||
| 113 | std::vector<u8> message; | ||
| 114 | std::vector<u8> addr; | ||
| 115 | s32 ret{}; | ||
| 116 | Errno bsd_errno{}; | ||
| 117 | }; | ||
| 118 | |||
| 18 | void RegisterClient(Kernel::HLERequestContext& ctx); | 119 | void RegisterClient(Kernel::HLERequestContext& ctx); |
| 19 | void StartMonitoring(Kernel::HLERequestContext& ctx); | 120 | void StartMonitoring(Kernel::HLERequestContext& ctx); |
| 20 | void Socket(Kernel::HLERequestContext& ctx); | 121 | void Socket(Kernel::HLERequestContext& ctx); |
| 21 | void Select(Kernel::HLERequestContext& ctx); | 122 | void Select(Kernel::HLERequestContext& ctx); |
| 123 | void Poll(Kernel::HLERequestContext& ctx); | ||
| 124 | void Accept(Kernel::HLERequestContext& ctx); | ||
| 22 | void Bind(Kernel::HLERequestContext& ctx); | 125 | void Bind(Kernel::HLERequestContext& ctx); |
| 23 | void Connect(Kernel::HLERequestContext& ctx); | 126 | void Connect(Kernel::HLERequestContext& ctx); |
| 127 | void GetPeerName(Kernel::HLERequestContext& ctx); | ||
| 128 | void GetSockName(Kernel::HLERequestContext& ctx); | ||
| 24 | void Listen(Kernel::HLERequestContext& ctx); | 129 | void Listen(Kernel::HLERequestContext& ctx); |
| 130 | void Fcntl(Kernel::HLERequestContext& ctx); | ||
| 25 | void SetSockOpt(Kernel::HLERequestContext& ctx); | 131 | void SetSockOpt(Kernel::HLERequestContext& ctx); |
| 132 | void Shutdown(Kernel::HLERequestContext& ctx); | ||
| 133 | void Recv(Kernel::HLERequestContext& ctx); | ||
| 134 | void RecvFrom(Kernel::HLERequestContext& ctx); | ||
| 135 | void Send(Kernel::HLERequestContext& ctx); | ||
| 26 | void SendTo(Kernel::HLERequestContext& ctx); | 136 | void SendTo(Kernel::HLERequestContext& ctx); |
| 137 | void Write(Kernel::HLERequestContext& ctx); | ||
| 27 | void Close(Kernel::HLERequestContext& ctx); | 138 | void Close(Kernel::HLERequestContext& ctx); |
| 28 | 139 | ||
| 29 | /// Id to use for the next open file descriptor. | 140 | template <typename Work> |
| 30 | u32 next_fd = 1; | 141 | void ExecuteWork(Kernel::HLERequestContext& ctx, std::string_view sleep_reason, |
| 142 | bool is_blocking, Work work); | ||
| 143 | |||
| 144 | std::pair<s32, Errno> SocketImpl(Domain domain, Type type, Protocol protocol); | ||
| 145 | std::pair<s32, Errno> PollImpl(std::vector<u8>& write_buffer, std::vector<u8> read_buffer, | ||
| 146 | s32 nfds, s32 timeout); | ||
| 147 | std::pair<s32, Errno> AcceptImpl(s32 fd, std::vector<u8>& write_buffer); | ||
| 148 | Errno BindImpl(s32 fd, const std::vector<u8>& addr); | ||
| 149 | Errno ConnectImpl(s32 fd, const std::vector<u8>& addr); | ||
| 150 | Errno GetPeerNameImpl(s32 fd, std::vector<u8>& write_buffer); | ||
| 151 | Errno GetSockNameImpl(s32 fd, std::vector<u8>& write_buffer); | ||
| 152 | Errno ListenImpl(s32 fd, s32 backlog); | ||
| 153 | std::pair<s32, Errno> FcntlImpl(s32 fd, FcntlCmd cmd, s32 arg); | ||
| 154 | Errno SetSockOptImpl(s32 fd, u32 level, OptName optname, size_t optlen, const void* optval); | ||
| 155 | Errno ShutdownImpl(s32 fd, s32 how); | ||
| 156 | std::pair<s32, Errno> RecvImpl(s32 fd, u32 flags, std::vector<u8>& message); | ||
| 157 | std::pair<s32, Errno> RecvFromImpl(s32 fd, u32 flags, std::vector<u8>& message, | ||
| 158 | std::vector<u8>& addr); | ||
| 159 | std::pair<s32, Errno> SendImpl(s32 fd, u32 flags, const std::vector<u8>& message); | ||
| 160 | std::pair<s32, Errno> SendToImpl(s32 fd, u32 flags, const std::vector<u8>& message, | ||
| 161 | const std::vector<u8>& addr); | ||
| 162 | Errno CloseImpl(s32 fd); | ||
| 163 | |||
| 164 | s32 FindFreeFileDescriptorHandle() noexcept; | ||
| 165 | bool IsFileDescriptorValid(s32 fd) const noexcept; | ||
| 166 | bool IsBlockingSocket(s32 fd) const noexcept; | ||
| 167 | |||
| 168 | void BuildErrnoResponse(Kernel::HLERequestContext& ctx, Errno bsd_errno) const noexcept; | ||
| 169 | |||
| 170 | std::array<std::optional<FileDescriptor>, MAX_FD> file_descriptors; | ||
| 171 | |||
| 172 | BlockingWorkerPool<BSD, PollWork, AcceptWork, ConnectWork, RecvWork, RecvFromWork, SendWork, | ||
| 173 | SendToWork> | ||
| 174 | worker_pool; | ||
| 31 | }; | 175 | }; |
| 32 | 176 | ||
| 33 | class BSDCFG final : public ServiceFramework<BSDCFG> { | 177 | class BSDCFG final : public ServiceFramework<BSDCFG> { |
diff --git a/src/core/hle/service/sockets/sockets.cpp b/src/core/hle/service/sockets/sockets.cpp index 08d2d306a..1d27f7906 100644 --- a/src/core/hle/service/sockets/sockets.cpp +++ b/src/core/hle/service/sockets/sockets.cpp | |||
| @@ -10,9 +10,9 @@ | |||
| 10 | 10 | ||
| 11 | namespace Service::Sockets { | 11 | namespace Service::Sockets { |
| 12 | 12 | ||
| 13 | void InstallInterfaces(SM::ServiceManager& service_manager) { | 13 | void InstallInterfaces(SM::ServiceManager& service_manager, Core::System& system) { |
| 14 | std::make_shared<BSD>("bsd:s")->InstallAsService(service_manager); | 14 | std::make_shared<BSD>(system, "bsd:s")->InstallAsService(service_manager); |
| 15 | std::make_shared<BSD>("bsd:u")->InstallAsService(service_manager); | 15 | std::make_shared<BSD>(system, "bsd:u")->InstallAsService(service_manager); |
| 16 | std::make_shared<BSDCFG>()->InstallAsService(service_manager); | 16 | std::make_shared<BSDCFG>()->InstallAsService(service_manager); |
| 17 | 17 | ||
| 18 | std::make_shared<ETHC_C>()->InstallAsService(service_manager); | 18 | std::make_shared<ETHC_C>()->InstallAsService(service_manager); |
diff --git a/src/core/hle/service/sockets/sockets.h b/src/core/hle/service/sockets/sockets.h index ca8a6a7e0..89a410076 100644 --- a/src/core/hle/service/sockets/sockets.h +++ b/src/core/hle/service/sockets/sockets.h | |||
| @@ -4,11 +4,94 @@ | |||
| 4 | 4 | ||
| 5 | #pragma once | 5 | #pragma once |
| 6 | 6 | ||
| 7 | #include "common/common_types.h" | ||
| 7 | #include "core/hle/service/service.h" | 8 | #include "core/hle/service/service.h" |
| 8 | 9 | ||
| 10 | namespace Core { | ||
| 11 | class System; | ||
| 12 | } | ||
| 13 | |||
| 9 | namespace Service::Sockets { | 14 | namespace Service::Sockets { |
| 10 | 15 | ||
| 16 | enum class Errno : u32 { | ||
| 17 | SUCCESS = 0, | ||
| 18 | BADF = 9, | ||
| 19 | AGAIN = 11, | ||
| 20 | INVAL = 22, | ||
| 21 | MFILE = 24, | ||
| 22 | NOTCONN = 107, | ||
| 23 | }; | ||
| 24 | |||
| 25 | enum class Domain : u32 { | ||
| 26 | INET = 2, | ||
| 27 | }; | ||
| 28 | |||
| 29 | enum class Type : u32 { | ||
| 30 | STREAM = 1, | ||
| 31 | DGRAM = 2, | ||
| 32 | RAW = 3, | ||
| 33 | SEQPACKET = 5, | ||
| 34 | }; | ||
| 35 | |||
| 36 | enum class Protocol : u32 { | ||
| 37 | UNSPECIFIED = 0, | ||
| 38 | ICMP = 1, | ||
| 39 | TCP = 6, | ||
| 40 | UDP = 17, | ||
| 41 | }; | ||
| 42 | |||
| 43 | enum class OptName : u32 { | ||
| 44 | REUSEADDR = 0x4, | ||
| 45 | BROADCAST = 0x20, | ||
| 46 | LINGER = 0x80, | ||
| 47 | SNDBUF = 0x1001, | ||
| 48 | RCVBUF = 0x1002, | ||
| 49 | SNDTIMEO = 0x1005, | ||
| 50 | RCVTIMEO = 0x1006, | ||
| 51 | }; | ||
| 52 | |||
| 53 | enum class ShutdownHow : s32 { | ||
| 54 | RD = 0, | ||
| 55 | WR = 1, | ||
| 56 | RDWR = 2, | ||
| 57 | }; | ||
| 58 | |||
| 59 | enum class FcntlCmd : s32 { | ||
| 60 | GETFL = 3, | ||
| 61 | SETFL = 4, | ||
| 62 | }; | ||
| 63 | |||
| 64 | struct SockAddrIn { | ||
| 65 | u8 len; | ||
| 66 | u8 family; | ||
| 67 | u16 portno; | ||
| 68 | std::array<u8, 4> ip; | ||
| 69 | std::array<u8, 8> zeroes; | ||
| 70 | }; | ||
| 71 | |||
| 72 | struct PollFD { | ||
| 73 | s32 fd; | ||
| 74 | u16 events; | ||
| 75 | u16 revents; | ||
| 76 | }; | ||
| 77 | |||
| 78 | struct Linger { | ||
| 79 | u32 onoff; | ||
| 80 | u32 linger; | ||
| 81 | }; | ||
| 82 | |||
| 83 | constexpr u16 POLL_IN = 0x01; | ||
| 84 | constexpr u16 POLL_PRI = 0x02; | ||
| 85 | constexpr u16 POLL_OUT = 0x04; | ||
| 86 | constexpr u16 POLL_ERR = 0x08; | ||
| 87 | constexpr u16 POLL_HUP = 0x10; | ||
| 88 | constexpr u16 POLL_NVAL = 0x20; | ||
| 89 | |||
| 90 | constexpr u32 FLAG_MSG_DONTWAIT = 0x80; | ||
| 91 | |||
| 92 | constexpr u32 FLAG_O_NONBLOCK = 0x800; | ||
| 93 | |||
| 11 | /// Registers all Sockets services with the specified service manager. | 94 | /// Registers all Sockets services with the specified service manager. |
| 12 | void InstallInterfaces(SM::ServiceManager& service_manager); | 95 | void InstallInterfaces(SM::ServiceManager& service_manager, Core::System& system); |
| 13 | 96 | ||
| 14 | } // namespace Service::Sockets | 97 | } // namespace Service::Sockets |
diff --git a/src/core/hle/service/sockets/sockets_translate.cpp b/src/core/hle/service/sockets/sockets_translate.cpp new file mode 100644 index 000000000..2be8f642d --- /dev/null +++ b/src/core/hle/service/sockets/sockets_translate.cpp | |||
| @@ -0,0 +1,165 @@ | |||
| 1 | // Copyright 2020 yuzu emulator team | ||
| 2 | // Licensed under GPLv2 or any later version | ||
| 3 | // Refer to the license.txt file included. | ||
| 4 | |||
| 5 | #include <utility> | ||
| 6 | |||
| 7 | #include "common/assert.h" | ||
| 8 | #include "common/common_types.h" | ||
| 9 | #include "core/hle/service/sockets/sockets.h" | ||
| 10 | #include "core/hle/service/sockets/sockets_translate.h" | ||
| 11 | #include "core/network/network.h" | ||
| 12 | |||
| 13 | namespace Service::Sockets { | ||
| 14 | |||
| 15 | Errno Translate(Network::Errno value) { | ||
| 16 | switch (value) { | ||
| 17 | case Network::Errno::SUCCESS: | ||
| 18 | return Errno::SUCCESS; | ||
| 19 | case Network::Errno::BADF: | ||
| 20 | return Errno::BADF; | ||
| 21 | case Network::Errno::AGAIN: | ||
| 22 | return Errno::AGAIN; | ||
| 23 | case Network::Errno::INVAL: | ||
| 24 | return Errno::INVAL; | ||
| 25 | case Network::Errno::MFILE: | ||
| 26 | return Errno::MFILE; | ||
| 27 | case Network::Errno::NOTCONN: | ||
| 28 | return Errno::NOTCONN; | ||
| 29 | default: | ||
| 30 | UNIMPLEMENTED_MSG("Unimplemented errno={}", static_cast<int>(value)); | ||
| 31 | return Errno::SUCCESS; | ||
| 32 | } | ||
| 33 | } | ||
| 34 | |||
| 35 | std::pair<s32, Errno> Translate(std::pair<s32, Network::Errno> value) { | ||
| 36 | return {value.first, Translate(value.second)}; | ||
| 37 | } | ||
| 38 | |||
| 39 | Network::Domain Translate(Domain domain) { | ||
| 40 | switch (domain) { | ||
| 41 | case Domain::INET: | ||
| 42 | return Network::Domain::INET; | ||
| 43 | default: | ||
| 44 | UNIMPLEMENTED_MSG("Unimplemented domain={}", static_cast<int>(domain)); | ||
| 45 | return {}; | ||
| 46 | } | ||
| 47 | } | ||
| 48 | |||
| 49 | Domain Translate(Network::Domain domain) { | ||
| 50 | switch (domain) { | ||
| 51 | case Network::Domain::INET: | ||
| 52 | return Domain::INET; | ||
| 53 | default: | ||
| 54 | UNIMPLEMENTED_MSG("Unimplemented domain={}", static_cast<int>(domain)); | ||
| 55 | return {}; | ||
| 56 | } | ||
| 57 | } | ||
| 58 | |||
| 59 | Network::Type Translate(Type type) { | ||
| 60 | switch (type) { | ||
| 61 | case Type::STREAM: | ||
| 62 | return Network::Type::STREAM; | ||
| 63 | case Type::DGRAM: | ||
| 64 | return Network::Type::DGRAM; | ||
| 65 | default: | ||
| 66 | UNIMPLEMENTED_MSG("Unimplemented type={}", static_cast<int>(type)); | ||
| 67 | } | ||
| 68 | } | ||
| 69 | |||
| 70 | Network::Protocol Translate(Type type, Protocol protocol) { | ||
| 71 | switch (protocol) { | ||
| 72 | case Protocol::UNSPECIFIED: | ||
| 73 | LOG_WARNING(Service, "Unspecified protocol, assuming protocol from type"); | ||
| 74 | switch (type) { | ||
| 75 | case Type::DGRAM: | ||
| 76 | return Network::Protocol::UDP; | ||
| 77 | case Type::STREAM: | ||
| 78 | return Network::Protocol::TCP; | ||
| 79 | default: | ||
| 80 | return Network::Protocol::TCP; | ||
| 81 | } | ||
| 82 | case Protocol::TCP: | ||
| 83 | return Network::Protocol::TCP; | ||
| 84 | case Protocol::UDP: | ||
| 85 | return Network::Protocol::UDP; | ||
| 86 | default: | ||
| 87 | UNIMPLEMENTED_MSG("Unimplemented protocol={}", static_cast<int>(protocol)); | ||
| 88 | return Network::Protocol::TCP; | ||
| 89 | } | ||
| 90 | } | ||
| 91 | |||
| 92 | u16 TranslatePollEventsToHost(u16 flags) { | ||
| 93 | u16 result = 0; | ||
| 94 | const auto translate = [&result, &flags](u16 from, u16 to) { | ||
| 95 | if ((flags & from) != 0) { | ||
| 96 | flags &= ~from; | ||
| 97 | result |= to; | ||
| 98 | } | ||
| 99 | }; | ||
| 100 | translate(POLL_IN, Network::POLL_IN); | ||
| 101 | translate(POLL_PRI, Network::POLL_PRI); | ||
| 102 | translate(POLL_OUT, Network::POLL_OUT); | ||
| 103 | translate(POLL_ERR, Network::POLL_ERR); | ||
| 104 | translate(POLL_HUP, Network::POLL_HUP); | ||
| 105 | translate(POLL_NVAL, Network::POLL_NVAL); | ||
| 106 | |||
| 107 | UNIMPLEMENTED_IF_MSG(flags != 0, "Unimplemented flags={}", flags); | ||
| 108 | return result; | ||
| 109 | } | ||
| 110 | |||
| 111 | u16 TranslatePollEventsToGuest(u16 flags) { | ||
| 112 | u16 result = 0; | ||
| 113 | const auto translate = [&result, &flags](u16 from, u16 to) { | ||
| 114 | if ((flags & from) != 0) { | ||
| 115 | flags &= ~from; | ||
| 116 | result |= to; | ||
| 117 | } | ||
| 118 | }; | ||
| 119 | |||
| 120 | translate(Network::POLL_IN, POLL_IN); | ||
| 121 | translate(Network::POLL_PRI, POLL_PRI); | ||
| 122 | translate(Network::POLL_OUT, POLL_OUT); | ||
| 123 | translate(Network::POLL_ERR, POLL_ERR); | ||
| 124 | translate(Network::POLL_HUP, POLL_HUP); | ||
| 125 | translate(Network::POLL_NVAL, POLL_NVAL); | ||
| 126 | |||
| 127 | UNIMPLEMENTED_IF_MSG(flags != 0, "Unimplemented flags={}", flags); | ||
| 128 | return result; | ||
| 129 | } | ||
| 130 | |||
| 131 | Network::SockAddrIn Translate(SockAddrIn value) { | ||
| 132 | ASSERT(value.len == 0 || value.len == sizeof(value)); | ||
| 133 | |||
| 134 | Network::SockAddrIn result; | ||
| 135 | result.family = Translate(static_cast<Domain>(value.family)); | ||
| 136 | result.ip = value.ip; | ||
| 137 | result.portno = value.portno >> 8 | value.portno << 8; | ||
| 138 | return result; | ||
| 139 | } | ||
| 140 | |||
| 141 | SockAddrIn Translate(Network::SockAddrIn value) { | ||
| 142 | SockAddrIn result; | ||
| 143 | result.len = sizeof(result); | ||
| 144 | result.family = static_cast<u8>(Translate(value.family)); | ||
| 145 | result.portno = value.portno >> 8 | value.portno << 8; | ||
| 146 | result.ip = value.ip; | ||
| 147 | result.zeroes = {}; | ||
| 148 | return result; | ||
| 149 | } | ||
| 150 | |||
| 151 | Network::ShutdownHow Translate(ShutdownHow how) { | ||
| 152 | switch (how) { | ||
| 153 | case ShutdownHow::RD: | ||
| 154 | return Network::ShutdownHow::RD; | ||
| 155 | case ShutdownHow::WR: | ||
| 156 | return Network::ShutdownHow::WR; | ||
| 157 | case ShutdownHow::RDWR: | ||
| 158 | return Network::ShutdownHow::RDWR; | ||
| 159 | default: | ||
| 160 | UNIMPLEMENTED_MSG("Unimplemented how={}", static_cast<int>(how)); | ||
| 161 | return {}; | ||
| 162 | } | ||
| 163 | } | ||
| 164 | |||
| 165 | } // namespace Service::Sockets | ||
diff --git a/src/core/hle/service/sockets/sockets_translate.h b/src/core/hle/service/sockets/sockets_translate.h new file mode 100644 index 000000000..8ed041e31 --- /dev/null +++ b/src/core/hle/service/sockets/sockets_translate.h | |||
| @@ -0,0 +1,48 @@ | |||
| 1 | // Copyright 2020 yuzu emulator team | ||
| 2 | // Licensed under GPLv2 or any later version | ||
| 3 | // Refer to the license.txt file included. | ||
| 4 | |||
| 5 | #pragma once | ||
| 6 | |||
| 7 | #include <utility> | ||
| 8 | |||
| 9 | #include "common/common_types.h" | ||
| 10 | #include "core/hle/service/sockets/sockets.h" | ||
| 11 | #include "core/network/network.h" | ||
| 12 | |||
| 13 | namespace Service::Sockets { | ||
| 14 | |||
| 15 | /// Translate abstract errno to guest errno | ||
| 16 | Errno Translate(Network::Errno value); | ||
| 17 | |||
| 18 | /// Translate abstract return value errno pair to guest return value errno pair | ||
| 19 | std::pair<s32, Errno> Translate(std::pair<s32, Network::Errno> value); | ||
| 20 | |||
| 21 | /// Translate guest domain to abstract domain | ||
| 22 | Network::Domain Translate(Domain domain); | ||
| 23 | |||
| 24 | /// Translate abstract domain to guest domain | ||
| 25 | Domain Translate(Network::Domain domain); | ||
| 26 | |||
| 27 | /// Translate guest type to abstract type | ||
| 28 | Network::Type Translate(Type type); | ||
| 29 | |||
| 30 | /// Translate guest protocol to abstract protocol | ||
| 31 | Network::Protocol Translate(Type type, Protocol protocol); | ||
| 32 | |||
| 33 | /// Translate abstract poll event flags to guest poll event flags | ||
| 34 | u16 TranslatePollEventsToHost(u16 flags); | ||
| 35 | |||
| 36 | /// Translate guest poll event flags to abstract poll event flags | ||
| 37 | u16 TranslatePollEventsToGuest(u16 flags); | ||
| 38 | |||
| 39 | /// Translate guest socket address structure to abstract socket address structure | ||
| 40 | Network::SockAddrIn Translate(SockAddrIn value); | ||
| 41 | |||
| 42 | /// Translate abstract socket address structure to guest socket address structure | ||
| 43 | SockAddrIn Translate(Network::SockAddrIn value); | ||
| 44 | |||
| 45 | /// Translate guest shutdown mode to abstract shutdown mode | ||
| 46 | Network::ShutdownHow Translate(ShutdownHow how); | ||
| 47 | |||
| 48 | } // namespace Service::Sockets | ||
diff --git a/src/core/settings.h b/src/core/settings.h index 732c6a894..80f0d95a7 100644 --- a/src/core/settings.h +++ b/src/core/settings.h | |||
| @@ -67,6 +67,11 @@ private: | |||
| 67 | Type local{}; | 67 | Type local{}; |
| 68 | }; | 68 | }; |
| 69 | 69 | ||
| 70 | struct TouchFromButtonMap { | ||
| 71 | std::string name; | ||
| 72 | std::vector<std::string> buttons; | ||
| 73 | }; | ||
| 74 | |||
| 70 | struct Values { | 75 | struct Values { |
| 71 | // Audio | 76 | // Audio |
| 72 | std::string audio_device_id; | 77 | std::string audio_device_id; |
| @@ -145,15 +150,18 @@ struct Values { | |||
| 145 | ButtonsRaw debug_pad_buttons; | 150 | ButtonsRaw debug_pad_buttons; |
| 146 | AnalogsRaw debug_pad_analogs; | 151 | AnalogsRaw debug_pad_analogs; |
| 147 | 152 | ||
| 148 | std::string motion_device; | ||
| 149 | |||
| 150 | bool vibration_enabled; | 153 | bool vibration_enabled; |
| 151 | 154 | ||
| 155 | std::string motion_device; | ||
| 156 | std::string touch_device; | ||
| 152 | TouchscreenInput touchscreen; | 157 | TouchscreenInput touchscreen; |
| 153 | std::atomic_bool is_device_reload_pending{true}; | 158 | std::atomic_bool is_device_reload_pending{true}; |
| 159 | bool use_touch_from_button; | ||
| 160 | int touch_from_button_map_index; | ||
| 154 | std::string udp_input_address; | 161 | std::string udp_input_address; |
| 155 | u16 udp_input_port; | 162 | u16 udp_input_port; |
| 156 | u8 udp_pad_index; | 163 | u8 udp_pad_index; |
| 164 | std::vector<TouchFromButtonMap> touch_from_button_maps; | ||
| 157 | 165 | ||
| 158 | // Data Storage | 166 | // Data Storage |
| 159 | bool use_virtual_sd; | 167 | bool use_virtual_sd; |
diff --git a/src/input_common/CMakeLists.txt b/src/input_common/CMakeLists.txt index 56267c8a8..09361e37e 100644 --- a/src/input_common/CMakeLists.txt +++ b/src/input_common/CMakeLists.txt | |||
| @@ -7,8 +7,12 @@ add_library(input_common STATIC | |||
| 7 | main.h | 7 | main.h |
| 8 | motion_emu.cpp | 8 | motion_emu.cpp |
| 9 | motion_emu.h | 9 | motion_emu.h |
| 10 | motion_input.cpp | ||
| 11 | motion_input.h | ||
| 10 | settings.cpp | 12 | settings.cpp |
| 11 | settings.h | 13 | settings.h |
| 14 | touch_from_button.cpp | ||
| 15 | touch_from_button.h | ||
| 12 | gcadapter/gc_adapter.cpp | 16 | gcadapter/gc_adapter.cpp |
| 13 | gcadapter/gc_adapter.h | 17 | gcadapter/gc_adapter.h |
| 14 | gcadapter/gc_poller.cpp | 18 | gcadapter/gc_poller.cpp |
diff --git a/src/input_common/gcadapter/gc_poller.cpp b/src/input_common/gcadapter/gc_poller.cpp index 71cd85eeb..1c8d8523a 100644 --- a/src/input_common/gcadapter/gc_poller.cpp +++ b/src/input_common/gcadapter/gc_poller.cpp | |||
| @@ -38,7 +38,8 @@ public: | |||
| 38 | explicit GCAxisButton(int port_, int axis_, float threshold_, bool trigger_if_greater_, | 38 | explicit GCAxisButton(int port_, int axis_, float threshold_, bool trigger_if_greater_, |
| 39 | GCAdapter::Adapter* adapter) | 39 | GCAdapter::Adapter* adapter) |
| 40 | : port(port_), axis(axis_), threshold(threshold_), trigger_if_greater(trigger_if_greater_), | 40 | : port(port_), axis(axis_), threshold(threshold_), trigger_if_greater(trigger_if_greater_), |
| 41 | gcadapter(adapter), origin_value(adapter->GetOriginValue(port_, axis_)) {} | 41 | gcadapter(adapter), |
| 42 | origin_value(static_cast<float>(adapter->GetOriginValue(port_, axis_))) {} | ||
| 42 | 43 | ||
| 43 | bool GetStatus() const override { | 44 | bool GetStatus() const override { |
| 44 | if (gcadapter->DeviceConnected(port)) { | 45 | if (gcadapter->DeviceConnected(port)) { |
| @@ -151,8 +152,9 @@ public: | |||
| 151 | GCAnalog(int port_, int axis_x_, int axis_y_, float deadzone_, GCAdapter::Adapter* adapter, | 152 | GCAnalog(int port_, int axis_x_, int axis_y_, float deadzone_, GCAdapter::Adapter* adapter, |
| 152 | float range_) | 153 | float range_) |
| 153 | : port(port_), axis_x(axis_x_), axis_y(axis_y_), deadzone(deadzone_), gcadapter(adapter), | 154 | : port(port_), axis_x(axis_x_), axis_y(axis_y_), deadzone(deadzone_), gcadapter(adapter), |
| 154 | origin_value_x(adapter->GetOriginValue(port_, axis_x_)), | 155 | origin_value_x(static_cast<float>(adapter->GetOriginValue(port_, axis_x_))), |
| 155 | origin_value_y(adapter->GetOriginValue(port_, axis_y_)), range(range_) {} | 156 | origin_value_y(static_cast<float>(adapter->GetOriginValue(port_, axis_y_))), |
| 157 | range(range_) {} | ||
| 156 | 158 | ||
| 157 | float GetAxis(int axis) const { | 159 | float GetAxis(int axis) const { |
| 158 | if (gcadapter->DeviceConnected(port)) { | 160 | if (gcadapter->DeviceConnected(port)) { |
diff --git a/src/input_common/main.cpp b/src/input_common/main.cpp index 57e7a25fe..ea1a1cee6 100644 --- a/src/input_common/main.cpp +++ b/src/input_common/main.cpp | |||
| @@ -11,6 +11,7 @@ | |||
| 11 | #include "input_common/keyboard.h" | 11 | #include "input_common/keyboard.h" |
| 12 | #include "input_common/main.h" | 12 | #include "input_common/main.h" |
| 13 | #include "input_common/motion_emu.h" | 13 | #include "input_common/motion_emu.h" |
| 14 | #include "input_common/touch_from_button.h" | ||
| 14 | #include "input_common/udp/udp.h" | 15 | #include "input_common/udp/udp.h" |
| 15 | #ifdef HAVE_SDL2 | 16 | #ifdef HAVE_SDL2 |
| 16 | #include "input_common/sdl/sdl.h" | 17 | #include "input_common/sdl/sdl.h" |
| @@ -32,6 +33,8 @@ struct InputSubsystem::Impl { | |||
| 32 | std::make_shared<AnalogFromButton>()); | 33 | std::make_shared<AnalogFromButton>()); |
| 33 | motion_emu = std::make_shared<MotionEmu>(); | 34 | motion_emu = std::make_shared<MotionEmu>(); |
| 34 | Input::RegisterFactory<Input::MotionDevice>("motion_emu", motion_emu); | 35 | Input::RegisterFactory<Input::MotionDevice>("motion_emu", motion_emu); |
| 36 | Input::RegisterFactory<Input::TouchDevice>("touch_from_button", | ||
| 37 | std::make_shared<TouchFromButtonFactory>()); | ||
| 35 | 38 | ||
| 36 | #ifdef HAVE_SDL2 | 39 | #ifdef HAVE_SDL2 |
| 37 | sdl = SDL::Init(); | 40 | sdl = SDL::Init(); |
| @@ -46,6 +49,7 @@ struct InputSubsystem::Impl { | |||
| 46 | Input::UnregisterFactory<Input::AnalogDevice>("analog_from_button"); | 49 | Input::UnregisterFactory<Input::AnalogDevice>("analog_from_button"); |
| 47 | Input::UnregisterFactory<Input::MotionDevice>("motion_emu"); | 50 | Input::UnregisterFactory<Input::MotionDevice>("motion_emu"); |
| 48 | motion_emu.reset(); | 51 | motion_emu.reset(); |
| 52 | Input::UnregisterFactory<Input::TouchDevice>("touch_from_button"); | ||
| 49 | #ifdef HAVE_SDL2 | 53 | #ifdef HAVE_SDL2 |
| 50 | sdl.reset(); | 54 | sdl.reset(); |
| 51 | #endif | 55 | #endif |
| @@ -171,6 +175,13 @@ const GCButtonFactory* InputSubsystem::GetGCButtons() const { | |||
| 171 | return impl->gcbuttons.get(); | 175 | return impl->gcbuttons.get(); |
| 172 | } | 176 | } |
| 173 | 177 | ||
| 178 | void InputSubsystem::ReloadInputDevices() { | ||
| 179 | if (!impl->udp) { | ||
| 180 | return; | ||
| 181 | } | ||
| 182 | impl->udp->ReloadUDPClient(); | ||
| 183 | } | ||
| 184 | |||
| 174 | std::vector<std::unique_ptr<Polling::DevicePoller>> InputSubsystem::GetPollers( | 185 | std::vector<std::unique_ptr<Polling::DevicePoller>> InputSubsystem::GetPollers( |
| 175 | Polling::DeviceType type) const { | 186 | Polling::DeviceType type) const { |
| 176 | #ifdef HAVE_SDL2 | 187 | #ifdef HAVE_SDL2 |
diff --git a/src/input_common/main.h b/src/input_common/main.h index 58e5dc250..f3fbf696e 100644 --- a/src/input_common/main.h +++ b/src/input_common/main.h | |||
| @@ -115,6 +115,9 @@ public: | |||
| 115 | /// Retrieves the underlying GameCube button handler. | 115 | /// Retrieves the underlying GameCube button handler. |
| 116 | [[nodiscard]] const GCButtonFactory* GetGCButtons() const; | 116 | [[nodiscard]] const GCButtonFactory* GetGCButtons() const; |
| 117 | 117 | ||
| 118 | /// Reloads the input devices | ||
| 119 | void ReloadInputDevices(); | ||
| 120 | |||
| 118 | /// Get all DevicePoller from all backends for a specific device type | 121 | /// Get all DevicePoller from all backends for a specific device type |
| 119 | [[nodiscard]] std::vector<std::unique_ptr<Polling::DevicePoller>> GetPollers( | 122 | [[nodiscard]] std::vector<std::unique_ptr<Polling::DevicePoller>> GetPollers( |
| 120 | Polling::DeviceType type) const; | 123 | Polling::DeviceType type) const; |
diff --git a/src/input_common/motion_input.cpp b/src/input_common/motion_input.cpp new file mode 100644 index 000000000..22a849866 --- /dev/null +++ b/src/input_common/motion_input.cpp | |||
| @@ -0,0 +1,181 @@ | |||
| 1 | // Copyright 2020 yuzu Emulator Project | ||
| 2 | // Licensed under GPLv2 or any later version | ||
| 3 | // Refer to the license.txt file included | ||
| 4 | |||
| 5 | #include "common/math_util.h" | ||
| 6 | #include "input_common/motion_input.h" | ||
| 7 | |||
| 8 | namespace InputCommon { | ||
| 9 | |||
| 10 | MotionInput::MotionInput(f32 new_kp, f32 new_ki, f32 new_kd) | ||
| 11 | : kp(new_kp), ki(new_ki), kd(new_kd), quat{{0, 0, -1}, 0} {} | ||
| 12 | |||
| 13 | void MotionInput::SetAcceleration(const Common::Vec3f& acceleration) { | ||
| 14 | accel = acceleration; | ||
| 15 | } | ||
| 16 | |||
| 17 | void MotionInput::SetGyroscope(const Common::Vec3f& gyroscope) { | ||
| 18 | gyro = gyroscope - gyro_drift; | ||
| 19 | if (gyro.Length2() < gyro_threshold) { | ||
| 20 | gyro = {}; | ||
| 21 | } | ||
| 22 | } | ||
| 23 | |||
| 24 | void MotionInput::SetQuaternion(const Common::Quaternion<f32>& quaternion) { | ||
| 25 | quat = quaternion; | ||
| 26 | } | ||
| 27 | |||
| 28 | void MotionInput::SetGyroDrift(const Common::Vec3f& drift) { | ||
| 29 | gyro_drift = drift; | ||
| 30 | } | ||
| 31 | |||
| 32 | void MotionInput::SetGyroThreshold(f32 threshold) { | ||
| 33 | gyro_threshold = threshold; | ||
| 34 | } | ||
| 35 | |||
| 36 | void MotionInput::EnableReset(bool reset) { | ||
| 37 | reset_enabled = reset; | ||
| 38 | } | ||
| 39 | |||
| 40 | void MotionInput::ResetRotations() { | ||
| 41 | rotations = {}; | ||
| 42 | } | ||
| 43 | |||
| 44 | bool MotionInput::IsMoving(f32 sensitivity) const { | ||
| 45 | return gyro.Length() >= sensitivity || accel.Length() <= 0.9f || accel.Length() >= 1.1f; | ||
| 46 | } | ||
| 47 | |||
| 48 | bool MotionInput::IsCalibrated(f32 sensitivity) const { | ||
| 49 | return real_error.Length() < sensitivity; | ||
| 50 | } | ||
| 51 | |||
| 52 | void MotionInput::UpdateRotation(u64 elapsed_time) { | ||
| 53 | const f32 sample_period = elapsed_time / 1000000.0f; | ||
| 54 | if (sample_period > 0.1f) { | ||
| 55 | return; | ||
| 56 | } | ||
| 57 | rotations += gyro * sample_period; | ||
| 58 | } | ||
| 59 | |||
| 60 | void MotionInput::UpdateOrientation(u64 elapsed_time) { | ||
| 61 | if (!IsCalibrated(0.1f)) { | ||
| 62 | ResetOrientation(); | ||
| 63 | } | ||
| 64 | // Short name local variable for readability | ||
| 65 | f32 q1 = quat.w; | ||
| 66 | f32 q2 = quat.xyz[0]; | ||
| 67 | f32 q3 = quat.xyz[1]; | ||
| 68 | f32 q4 = quat.xyz[2]; | ||
| 69 | const f32 sample_period = elapsed_time / 1000000.0f; | ||
| 70 | |||
| 71 | // ignore invalid elapsed time | ||
| 72 | if (sample_period > 0.1f) { | ||
| 73 | return; | ||
| 74 | } | ||
| 75 | |||
| 76 | const auto normal_accel = accel.Normalized(); | ||
| 77 | auto rad_gyro = gyro * Common::PI * 2; | ||
| 78 | const f32 swap = rad_gyro.x; | ||
| 79 | rad_gyro.x = rad_gyro.y; | ||
| 80 | rad_gyro.y = -swap; | ||
| 81 | rad_gyro.z = -rad_gyro.z; | ||
| 82 | |||
| 83 | // Ignore drift correction if acceleration is not reliable | ||
| 84 | if (accel.Length() >= 0.75f && accel.Length() <= 1.25f) { | ||
| 85 | const f32 ax = -normal_accel.x; | ||
| 86 | const f32 ay = normal_accel.y; | ||
| 87 | const f32 az = -normal_accel.z; | ||
| 88 | |||
| 89 | // Estimated direction of gravity | ||
| 90 | const f32 vx = 2.0f * (q2 * q4 - q1 * q3); | ||
| 91 | const f32 vy = 2.0f * (q1 * q2 + q3 * q4); | ||
| 92 | const f32 vz = q1 * q1 - q2 * q2 - q3 * q3 + q4 * q4; | ||
| 93 | |||
| 94 | // Error is cross product between estimated direction and measured direction of gravity | ||
| 95 | const Common::Vec3f new_real_error = {az * vx - ax * vz, ay * vz - az * vy, | ||
| 96 | ax * vy - ay * vx}; | ||
| 97 | |||
| 98 | derivative_error = new_real_error - real_error; | ||
| 99 | real_error = new_real_error; | ||
| 100 | |||
| 101 | // Prevent integral windup | ||
| 102 | if (ki != 0.0f && !IsCalibrated(0.05f)) { | ||
| 103 | integral_error += real_error; | ||
| 104 | } else { | ||
| 105 | integral_error = {}; | ||
| 106 | } | ||
| 107 | |||
| 108 | // Apply feedback terms | ||
| 109 | rad_gyro += kp * real_error; | ||
| 110 | rad_gyro += ki * integral_error; | ||
| 111 | rad_gyro += kd * derivative_error; | ||
| 112 | } | ||
| 113 | |||
| 114 | const f32 gx = rad_gyro.y; | ||
| 115 | const f32 gy = rad_gyro.x; | ||
| 116 | const f32 gz = rad_gyro.z; | ||
| 117 | |||
| 118 | // Integrate rate of change of quaternion | ||
| 119 | const f32 pa = q2; | ||
| 120 | const f32 pb = q3; | ||
| 121 | const f32 pc = q4; | ||
| 122 | q1 = q1 + (-q2 * gx - q3 * gy - q4 * gz) * (0.5f * sample_period); | ||
| 123 | q2 = pa + (q1 * gx + pb * gz - pc * gy) * (0.5f * sample_period); | ||
| 124 | q3 = pb + (q1 * gy - pa * gz + pc * gx) * (0.5f * sample_period); | ||
| 125 | q4 = pc + (q1 * gz + pa * gy - pb * gx) * (0.5f * sample_period); | ||
| 126 | |||
| 127 | quat.w = q1; | ||
| 128 | quat.xyz[0] = q2; | ||
| 129 | quat.xyz[1] = q3; | ||
| 130 | quat.xyz[2] = q4; | ||
| 131 | quat = quat.Normalized(); | ||
| 132 | } | ||
| 133 | |||
| 134 | std::array<Common::Vec3f, 3> MotionInput::GetOrientation() const { | ||
| 135 | const Common::Quaternion<float> quad{ | ||
| 136 | .xyz = {-quat.xyz[1], -quat.xyz[0], -quat.w}, | ||
| 137 | .w = -quat.xyz[2], | ||
| 138 | }; | ||
| 139 | const std::array<float, 16> matrix4x4 = quad.ToMatrix(); | ||
| 140 | |||
| 141 | return {Common::Vec3f(matrix4x4[0], matrix4x4[1], -matrix4x4[2]), | ||
| 142 | Common::Vec3f(matrix4x4[4], matrix4x4[5], -matrix4x4[6]), | ||
| 143 | Common::Vec3f(-matrix4x4[8], -matrix4x4[9], matrix4x4[10])}; | ||
| 144 | } | ||
| 145 | |||
| 146 | Common::Vec3f MotionInput::GetAcceleration() const { | ||
| 147 | return accel; | ||
| 148 | } | ||
| 149 | |||
| 150 | Common::Vec3f MotionInput::GetGyroscope() const { | ||
| 151 | return gyro; | ||
| 152 | } | ||
| 153 | |||
| 154 | Common::Quaternion<f32> MotionInput::GetQuaternion() const { | ||
| 155 | return quat; | ||
| 156 | } | ||
| 157 | |||
| 158 | Common::Vec3f MotionInput::GetRotations() const { | ||
| 159 | return rotations; | ||
| 160 | } | ||
| 161 | |||
| 162 | void MotionInput::ResetOrientation() { | ||
| 163 | if (!reset_enabled) { | ||
| 164 | return; | ||
| 165 | } | ||
| 166 | if (!IsMoving(0.5f) && accel.z <= -0.9f) { | ||
| 167 | ++reset_counter; | ||
| 168 | if (reset_counter > 900) { | ||
| 169 | // TODO: calculate quaternion from gravity vector | ||
| 170 | quat.w = 0; | ||
| 171 | quat.xyz[0] = 0; | ||
| 172 | quat.xyz[1] = 0; | ||
| 173 | quat.xyz[2] = -1; | ||
| 174 | integral_error = {}; | ||
| 175 | reset_counter = 0; | ||
| 176 | } | ||
| 177 | } else { | ||
| 178 | reset_counter = 0; | ||
| 179 | } | ||
| 180 | } | ||
| 181 | } // namespace InputCommon | ||
diff --git a/src/input_common/motion_input.h b/src/input_common/motion_input.h new file mode 100644 index 000000000..54b4439d9 --- /dev/null +++ b/src/input_common/motion_input.h | |||
| @@ -0,0 +1,68 @@ | |||
| 1 | // Copyright 2020 yuzu Emulator Project | ||
| 2 | // Licensed under GPLv2 or any later version | ||
| 3 | // Refer to the license.txt file included | ||
| 4 | |||
| 5 | #pragma once | ||
| 6 | |||
| 7 | #include "common/common_types.h" | ||
| 8 | #include "common/quaternion.h" | ||
| 9 | #include "common/vector_math.h" | ||
| 10 | |||
| 11 | namespace InputCommon { | ||
| 12 | |||
| 13 | class MotionInput { | ||
| 14 | public: | ||
| 15 | MotionInput(f32 new_kp, f32 new_ki, f32 new_kd); | ||
| 16 | |||
| 17 | MotionInput(const MotionInput&) = default; | ||
| 18 | MotionInput& operator=(const MotionInput&) = default; | ||
| 19 | |||
| 20 | MotionInput(MotionInput&&) = default; | ||
| 21 | MotionInput& operator=(MotionInput&&) = default; | ||
| 22 | |||
| 23 | void SetAcceleration(const Common::Vec3f& acceleration); | ||
| 24 | void SetGyroscope(const Common::Vec3f& acceleration); | ||
| 25 | void SetQuaternion(const Common::Quaternion<f32>& quaternion); | ||
| 26 | void SetGyroDrift(const Common::Vec3f& drift); | ||
| 27 | void SetGyroThreshold(f32 threshold); | ||
| 28 | |||
| 29 | void EnableReset(bool reset); | ||
| 30 | void ResetRotations(); | ||
| 31 | |||
| 32 | void UpdateRotation(u64 elapsed_time); | ||
| 33 | void UpdateOrientation(u64 elapsed_time); | ||
| 34 | |||
| 35 | std::array<Common::Vec3f, 3> GetOrientation() const; | ||
| 36 | Common::Vec3f GetAcceleration() const; | ||
| 37 | Common::Vec3f GetGyroscope() const; | ||
| 38 | Common::Vec3f GetRotations() const; | ||
| 39 | Common::Quaternion<f32> GetQuaternion() const; | ||
| 40 | |||
| 41 | bool IsMoving(f32 sensitivity) const; | ||
| 42 | bool IsCalibrated(f32 sensitivity) const; | ||
| 43 | |||
| 44 | private: | ||
| 45 | void ResetOrientation(); | ||
| 46 | |||
| 47 | // PID constants | ||
| 48 | const f32 kp; | ||
| 49 | const f32 ki; | ||
| 50 | const f32 kd; | ||
| 51 | |||
| 52 | // PID errors | ||
| 53 | Common::Vec3f real_error; | ||
| 54 | Common::Vec3f integral_error; | ||
| 55 | Common::Vec3f derivative_error; | ||
| 56 | |||
| 57 | Common::Quaternion<f32> quat; | ||
| 58 | Common::Vec3f rotations; | ||
| 59 | Common::Vec3f accel; | ||
| 60 | Common::Vec3f gyro; | ||
| 61 | Common::Vec3f gyro_drift; | ||
| 62 | |||
| 63 | f32 gyro_threshold = 0.0f; | ||
| 64 | u32 reset_counter = 0; | ||
| 65 | bool reset_enabled = true; | ||
| 66 | }; | ||
| 67 | |||
| 68 | } // namespace InputCommon | ||
diff --git a/src/input_common/sdl/sdl_impl.cpp b/src/input_common/sdl/sdl_impl.cpp index c8d9eb2bc..a9e676f4b 100644 --- a/src/input_common/sdl/sdl_impl.cpp +++ b/src/input_common/sdl/sdl_impl.cpp | |||
| @@ -3,6 +3,7 @@ | |||
| 3 | // Refer to the license.txt file included. | 3 | // Refer to the license.txt file included. |
| 4 | 4 | ||
| 5 | #include <algorithm> | 5 | #include <algorithm> |
| 6 | #include <array> | ||
| 6 | #include <atomic> | 7 | #include <atomic> |
| 7 | #include <cmath> | 8 | #include <cmath> |
| 8 | #include <functional> | 9 | #include <functional> |
| @@ -358,7 +359,7 @@ public: | |||
| 358 | return std::make_tuple(x / r * (r - deadzone) / (1 - deadzone), | 359 | return std::make_tuple(x / r * (r - deadzone) / (1 - deadzone), |
| 359 | y / r * (r - deadzone) / (1 - deadzone)); | 360 | y / r * (r - deadzone) / (1 - deadzone)); |
| 360 | } | 361 | } |
| 361 | return std::make_tuple<float, float>(0.0f, 0.0f); | 362 | return {}; |
| 362 | } | 363 | } |
| 363 | 364 | ||
| 364 | bool GetAnalogDirectionStatus(Input::AnalogDirection direction) const override { | 365 | bool GetAnalogDirectionStatus(Input::AnalogDirection direction) const override { |
| @@ -574,10 +575,10 @@ std::vector<Common::ParamPackage> SDLState::GetInputDevices() { | |||
| 574 | 575 | ||
| 575 | namespace { | 576 | namespace { |
| 576 | Common::ParamPackage BuildAnalogParamPackageForButton(int port, std::string guid, u8 axis, | 577 | Common::ParamPackage BuildAnalogParamPackageForButton(int port, std::string guid, u8 axis, |
| 577 | float value = 0.1) { | 578 | float value = 0.1f) { |
| 578 | Common::ParamPackage params({{"engine", "sdl"}}); | 579 | Common::ParamPackage params({{"engine", "sdl"}}); |
| 579 | params.Set("port", port); | 580 | params.Set("port", port); |
| 580 | params.Set("guid", guid); | 581 | params.Set("guid", std::move(guid)); |
| 581 | params.Set("axis", axis); | 582 | params.Set("axis", axis); |
| 582 | if (value > 0) { | 583 | if (value > 0) { |
| 583 | params.Set("direction", "+"); | 584 | params.Set("direction", "+"); |
| @@ -592,7 +593,7 @@ Common::ParamPackage BuildAnalogParamPackageForButton(int port, std::string guid | |||
| 592 | Common::ParamPackage BuildButtonParamPackageForButton(int port, std::string guid, u8 button) { | 593 | Common::ParamPackage BuildButtonParamPackageForButton(int port, std::string guid, u8 button) { |
| 593 | Common::ParamPackage params({{"engine", "sdl"}}); | 594 | Common::ParamPackage params({{"engine", "sdl"}}); |
| 594 | params.Set("port", port); | 595 | params.Set("port", port); |
| 595 | params.Set("guid", guid); | 596 | params.Set("guid", std::move(guid)); |
| 596 | params.Set("button", button); | 597 | params.Set("button", button); |
| 597 | return params; | 598 | return params; |
| 598 | } | 599 | } |
| @@ -601,7 +602,7 @@ Common::ParamPackage BuildHatParamPackageForButton(int port, std::string guid, u | |||
| 601 | Common::ParamPackage params({{"engine", "sdl"}}); | 602 | Common::ParamPackage params({{"engine", "sdl"}}); |
| 602 | 603 | ||
| 603 | params.Set("port", port); | 604 | params.Set("port", port); |
| 604 | params.Set("guid", guid); | 605 | params.Set("guid", std::move(guid)); |
| 605 | params.Set("hat", hat); | 606 | params.Set("hat", hat); |
| 606 | switch (value) { | 607 | switch (value) { |
| 607 | case SDL_HAT_UP: | 608 | case SDL_HAT_UP: |
| @@ -670,55 +671,62 @@ Common::ParamPackage BuildParamPackageForAnalog(int port, const std::string& gui | |||
| 670 | } // Anonymous namespace | 671 | } // Anonymous namespace |
| 671 | 672 | ||
| 672 | ButtonMapping SDLState::GetButtonMappingForDevice(const Common::ParamPackage& params) { | 673 | ButtonMapping SDLState::GetButtonMappingForDevice(const Common::ParamPackage& params) { |
| 673 | // This list is missing ZL/ZR since those are not considered buttons in SDL GameController. | ||
| 674 | // We will add those afterwards | ||
| 675 | // This list also excludes Screenshot since theres not really a mapping for that | ||
| 676 | std::unordered_map<Settings::NativeButton::Values, SDL_GameControllerButton> | ||
| 677 | switch_to_sdl_button = { | ||
| 678 | {Settings::NativeButton::A, SDL_CONTROLLER_BUTTON_B}, | ||
| 679 | {Settings::NativeButton::B, SDL_CONTROLLER_BUTTON_A}, | ||
| 680 | {Settings::NativeButton::X, SDL_CONTROLLER_BUTTON_Y}, | ||
| 681 | {Settings::NativeButton::Y, SDL_CONTROLLER_BUTTON_X}, | ||
| 682 | {Settings::NativeButton::LStick, SDL_CONTROLLER_BUTTON_LEFTSTICK}, | ||
| 683 | {Settings::NativeButton::RStick, SDL_CONTROLLER_BUTTON_RIGHTSTICK}, | ||
| 684 | {Settings::NativeButton::L, SDL_CONTROLLER_BUTTON_LEFTSHOULDER}, | ||
| 685 | {Settings::NativeButton::R, SDL_CONTROLLER_BUTTON_RIGHTSHOULDER}, | ||
| 686 | {Settings::NativeButton::Plus, SDL_CONTROLLER_BUTTON_START}, | ||
| 687 | {Settings::NativeButton::Minus, SDL_CONTROLLER_BUTTON_BACK}, | ||
| 688 | {Settings::NativeButton::DLeft, SDL_CONTROLLER_BUTTON_DPAD_LEFT}, | ||
| 689 | {Settings::NativeButton::DUp, SDL_CONTROLLER_BUTTON_DPAD_UP}, | ||
| 690 | {Settings::NativeButton::DRight, SDL_CONTROLLER_BUTTON_DPAD_RIGHT}, | ||
| 691 | {Settings::NativeButton::DDown, SDL_CONTROLLER_BUTTON_DPAD_DOWN}, | ||
| 692 | {Settings::NativeButton::SL, SDL_CONTROLLER_BUTTON_LEFTSHOULDER}, | ||
| 693 | {Settings::NativeButton::SR, SDL_CONTROLLER_BUTTON_RIGHTSHOULDER}, | ||
| 694 | {Settings::NativeButton::Home, SDL_CONTROLLER_BUTTON_GUIDE}, | ||
| 695 | }; | ||
| 696 | if (!params.Has("guid") || !params.Has("port")) { | 674 | if (!params.Has("guid") || !params.Has("port")) { |
| 697 | return {}; | 675 | return {}; |
| 698 | } | 676 | } |
| 699 | const auto joystick = GetSDLJoystickByGUID(params.Get("guid", ""), params.Get("port", 0)); | 677 | const auto joystick = GetSDLJoystickByGUID(params.Get("guid", ""), params.Get("port", 0)); |
| 700 | auto controller = joystick->GetSDLGameController(); | 678 | auto* controller = joystick->GetSDLGameController(); |
| 701 | if (!controller) { | 679 | if (controller == nullptr) { |
| 702 | return {}; | 680 | return {}; |
| 703 | } | 681 | } |
| 704 | 682 | ||
| 705 | ButtonMapping mapping{}; | 683 | // This list is missing ZL/ZR since those are not considered buttons in SDL GameController. |
| 684 | // We will add those afterwards | ||
| 685 | // This list also excludes Screenshot since theres not really a mapping for that | ||
| 686 | using ButtonBindings = | ||
| 687 | std::array<std::pair<Settings::NativeButton::Values, SDL_GameControllerButton>, 17>; | ||
| 688 | static constexpr ButtonBindings switch_to_sdl_button{{ | ||
| 689 | {Settings::NativeButton::A, SDL_CONTROLLER_BUTTON_B}, | ||
| 690 | {Settings::NativeButton::B, SDL_CONTROLLER_BUTTON_A}, | ||
| 691 | {Settings::NativeButton::X, SDL_CONTROLLER_BUTTON_Y}, | ||
| 692 | {Settings::NativeButton::Y, SDL_CONTROLLER_BUTTON_X}, | ||
| 693 | {Settings::NativeButton::LStick, SDL_CONTROLLER_BUTTON_LEFTSTICK}, | ||
| 694 | {Settings::NativeButton::RStick, SDL_CONTROLLER_BUTTON_RIGHTSTICK}, | ||
| 695 | {Settings::NativeButton::L, SDL_CONTROLLER_BUTTON_LEFTSHOULDER}, | ||
| 696 | {Settings::NativeButton::R, SDL_CONTROLLER_BUTTON_RIGHTSHOULDER}, | ||
| 697 | {Settings::NativeButton::Plus, SDL_CONTROLLER_BUTTON_START}, | ||
| 698 | {Settings::NativeButton::Minus, SDL_CONTROLLER_BUTTON_BACK}, | ||
| 699 | {Settings::NativeButton::DLeft, SDL_CONTROLLER_BUTTON_DPAD_LEFT}, | ||
| 700 | {Settings::NativeButton::DUp, SDL_CONTROLLER_BUTTON_DPAD_UP}, | ||
| 701 | {Settings::NativeButton::DRight, SDL_CONTROLLER_BUTTON_DPAD_RIGHT}, | ||
| 702 | {Settings::NativeButton::DDown, SDL_CONTROLLER_BUTTON_DPAD_DOWN}, | ||
| 703 | {Settings::NativeButton::SL, SDL_CONTROLLER_BUTTON_LEFTSHOULDER}, | ||
| 704 | {Settings::NativeButton::SR, SDL_CONTROLLER_BUTTON_RIGHTSHOULDER}, | ||
| 705 | {Settings::NativeButton::Home, SDL_CONTROLLER_BUTTON_GUIDE}, | ||
| 706 | }}; | ||
| 707 | |||
| 708 | // Add the missing bindings for ZL/ZR | ||
| 709 | using ZBindings = | ||
| 710 | std::array<std::pair<Settings::NativeButton::Values, SDL_GameControllerAxis>, 2>; | ||
| 711 | static constexpr ZBindings switch_to_sdl_axis{{ | ||
| 712 | {Settings::NativeButton::ZL, SDL_CONTROLLER_AXIS_TRIGGERLEFT}, | ||
| 713 | {Settings::NativeButton::ZR, SDL_CONTROLLER_AXIS_TRIGGERRIGHT}, | ||
| 714 | }}; | ||
| 715 | |||
| 716 | ButtonMapping mapping; | ||
| 717 | mapping.reserve(switch_to_sdl_button.size() + switch_to_sdl_axis.size()); | ||
| 718 | |||
| 706 | for (const auto& [switch_button, sdl_button] : switch_to_sdl_button) { | 719 | for (const auto& [switch_button, sdl_button] : switch_to_sdl_button) { |
| 707 | const auto& binding = SDL_GameControllerGetBindForButton(controller, sdl_button); | 720 | const auto& binding = SDL_GameControllerGetBindForButton(controller, sdl_button); |
| 708 | mapping[switch_button] = | 721 | mapping.insert_or_assign( |
| 709 | BuildParamPackageForBinding(joystick->GetPort(), joystick->GetGUID(), binding); | 722 | switch_button, |
| 723 | BuildParamPackageForBinding(joystick->GetPort(), joystick->GetGUID(), binding)); | ||
| 710 | } | 724 | } |
| 711 | |||
| 712 | // Add the missing bindings for ZL/ZR | ||
| 713 | std::unordered_map<Settings::NativeButton::Values, SDL_GameControllerAxis> switch_to_sdl_axis = | ||
| 714 | { | ||
| 715 | {Settings::NativeButton::ZL, SDL_CONTROLLER_AXIS_TRIGGERLEFT}, | ||
| 716 | {Settings::NativeButton::ZR, SDL_CONTROLLER_AXIS_TRIGGERRIGHT}, | ||
| 717 | }; | ||
| 718 | for (const auto& [switch_button, sdl_axis] : switch_to_sdl_axis) { | 725 | for (const auto& [switch_button, sdl_axis] : switch_to_sdl_axis) { |
| 719 | const auto& binding = SDL_GameControllerGetBindForAxis(controller, sdl_axis); | 726 | const auto& binding = SDL_GameControllerGetBindForAxis(controller, sdl_axis); |
| 720 | mapping[switch_button] = | 727 | mapping.insert_or_assign( |
| 721 | BuildParamPackageForBinding(joystick->GetPort(), joystick->GetGUID(), binding); | 728 | switch_button, |
| 729 | BuildParamPackageForBinding(joystick->GetPort(), joystick->GetGUID(), binding)); | ||
| 722 | } | 730 | } |
| 723 | 731 | ||
| 724 | return mapping; | 732 | return mapping; |
| @@ -729,8 +737,8 @@ AnalogMapping SDLState::GetAnalogMappingForDevice(const Common::ParamPackage& pa | |||
| 729 | return {}; | 737 | return {}; |
| 730 | } | 738 | } |
| 731 | const auto joystick = GetSDLJoystickByGUID(params.Get("guid", ""), params.Get("port", 0)); | 739 | const auto joystick = GetSDLJoystickByGUID(params.Get("guid", ""), params.Get("port", 0)); |
| 732 | auto controller = joystick->GetSDLGameController(); | 740 | auto* controller = joystick->GetSDLGameController(); |
| 733 | if (!controller) { | 741 | if (controller == nullptr) { |
| 734 | return {}; | 742 | return {}; |
| 735 | } | 743 | } |
| 736 | 744 | ||
| @@ -739,16 +747,18 @@ AnalogMapping SDLState::GetAnalogMappingForDevice(const Common::ParamPackage& pa | |||
| 739 | SDL_GameControllerGetBindForAxis(controller, SDL_CONTROLLER_AXIS_LEFTX); | 747 | SDL_GameControllerGetBindForAxis(controller, SDL_CONTROLLER_AXIS_LEFTX); |
| 740 | const auto& binding_left_y = | 748 | const auto& binding_left_y = |
| 741 | SDL_GameControllerGetBindForAxis(controller, SDL_CONTROLLER_AXIS_LEFTY); | 749 | SDL_GameControllerGetBindForAxis(controller, SDL_CONTROLLER_AXIS_LEFTY); |
| 742 | mapping[Settings::NativeAnalog::LStick] = | 750 | mapping.insert_or_assign(Settings::NativeAnalog::LStick, |
| 743 | BuildParamPackageForAnalog(joystick->GetPort(), joystick->GetGUID(), | 751 | BuildParamPackageForAnalog(joystick->GetPort(), joystick->GetGUID(), |
| 744 | binding_left_x.value.axis, binding_left_y.value.axis); | 752 | binding_left_x.value.axis, |
| 753 | binding_left_y.value.axis)); | ||
| 745 | const auto& binding_right_x = | 754 | const auto& binding_right_x = |
| 746 | SDL_GameControllerGetBindForAxis(controller, SDL_CONTROLLER_AXIS_RIGHTX); | 755 | SDL_GameControllerGetBindForAxis(controller, SDL_CONTROLLER_AXIS_RIGHTX); |
| 747 | const auto& binding_right_y = | 756 | const auto& binding_right_y = |
| 748 | SDL_GameControllerGetBindForAxis(controller, SDL_CONTROLLER_AXIS_RIGHTY); | 757 | SDL_GameControllerGetBindForAxis(controller, SDL_CONTROLLER_AXIS_RIGHTY); |
| 749 | mapping[Settings::NativeAnalog::RStick] = | 758 | mapping.insert_or_assign(Settings::NativeAnalog::RStick, |
| 750 | BuildParamPackageForAnalog(joystick->GetPort(), joystick->GetGUID(), | 759 | BuildParamPackageForAnalog(joystick->GetPort(), joystick->GetGUID(), |
| 751 | binding_right_x.value.axis, binding_right_y.value.axis); | 760 | binding_right_x.value.axis, |
| 761 | binding_right_y.value.axis)); | ||
| 752 | return mapping; | 762 | return mapping; |
| 753 | } | 763 | } |
| 754 | 764 | ||
| @@ -784,7 +794,7 @@ public: | |||
| 784 | } | 794 | } |
| 785 | return {}; | 795 | return {}; |
| 786 | } | 796 | } |
| 787 | std::optional<Common::ParamPackage> FromEvent(const SDL_Event& event) { | 797 | [[nodiscard]] std::optional<Common::ParamPackage> FromEvent(const SDL_Event& event) const { |
| 788 | switch (event.type) { | 798 | switch (event.type) { |
| 789 | case SDL_JOYAXISMOTION: | 799 | case SDL_JOYAXISMOTION: |
| 790 | if (std::abs(event.jaxis.value / 32767.0) < 0.5) { | 800 | if (std::abs(event.jaxis.value / 32767.0) < 0.5) { |
| @@ -795,7 +805,7 @@ public: | |||
| 795 | case SDL_JOYHATMOTION: | 805 | case SDL_JOYHATMOTION: |
| 796 | return {SDLEventToButtonParamPackage(state, event)}; | 806 | return {SDLEventToButtonParamPackage(state, event)}; |
| 797 | } | 807 | } |
| 798 | return {}; | 808 | return std::nullopt; |
| 799 | } | 809 | } |
| 800 | }; | 810 | }; |
| 801 | 811 | ||
diff --git a/src/input_common/touch_from_button.cpp b/src/input_common/touch_from_button.cpp new file mode 100644 index 000000000..98da0ef1a --- /dev/null +++ b/src/input_common/touch_from_button.cpp | |||
| @@ -0,0 +1,50 @@ | |||
| 1 | // Copyright 2020 Citra Emulator Project | ||
| 2 | // Licensed under GPLv2 or any later version | ||
| 3 | // Refer to the license.txt file included. | ||
| 4 | |||
| 5 | #include "core/frontend/framebuffer_layout.h" | ||
| 6 | #include "core/settings.h" | ||
| 7 | #include "input_common/touch_from_button.h" | ||
| 8 | |||
| 9 | namespace InputCommon { | ||
| 10 | |||
| 11 | class TouchFromButtonDevice final : public Input::TouchDevice { | ||
| 12 | public: | ||
| 13 | TouchFromButtonDevice() { | ||
| 14 | for (const auto& config_entry : | ||
| 15 | Settings::values.touch_from_button_maps[Settings::values.touch_from_button_map_index] | ||
| 16 | .buttons) { | ||
| 17 | const Common::ParamPackage package{config_entry}; | ||
| 18 | map.emplace_back( | ||
| 19 | Input::CreateDevice<Input::ButtonDevice>(config_entry), | ||
| 20 | std::clamp(package.Get("x", 0), 0, static_cast<int>(Layout::ScreenUndocked::Width)), | ||
| 21 | std::clamp(package.Get("y", 0), 0, | ||
| 22 | static_cast<int>(Layout::ScreenUndocked::Height))); | ||
| 23 | } | ||
| 24 | } | ||
| 25 | |||
| 26 | std::tuple<float, float, bool> GetStatus() const override { | ||
| 27 | for (const auto& m : map) { | ||
| 28 | const bool state = std::get<0>(m)->GetStatus(); | ||
| 29 | if (state) { | ||
| 30 | const float x = static_cast<float>(std::get<1>(m)) / | ||
| 31 | static_cast<int>(Layout::ScreenUndocked::Width); | ||
| 32 | const float y = static_cast<float>(std::get<2>(m)) / | ||
| 33 | static_cast<int>(Layout::ScreenUndocked::Height); | ||
| 34 | return {x, y, true}; | ||
| 35 | } | ||
| 36 | } | ||
| 37 | return {}; | ||
| 38 | } | ||
| 39 | |||
| 40 | private: | ||
| 41 | // A vector of the mapped button, its x and its y-coordinate | ||
| 42 | std::vector<std::tuple<std::unique_ptr<Input::ButtonDevice>, int, int>> map; | ||
| 43 | }; | ||
| 44 | |||
| 45 | std::unique_ptr<Input::TouchDevice> TouchFromButtonFactory::Create( | ||
| 46 | const Common::ParamPackage& params) { | ||
| 47 | return std::make_unique<TouchFromButtonDevice>(); | ||
| 48 | } | ||
| 49 | |||
| 50 | } // namespace InputCommon | ||
diff --git a/src/input_common/touch_from_button.h b/src/input_common/touch_from_button.h new file mode 100644 index 000000000..8b4d1aa96 --- /dev/null +++ b/src/input_common/touch_from_button.h | |||
| @@ -0,0 +1,23 @@ | |||
| 1 | // Copyright 2020 Citra Emulator Project | ||
| 2 | // Licensed under GPLv2 or any later version | ||
| 3 | // Refer to the license.txt file included. | ||
| 4 | |||
| 5 | #pragma once | ||
| 6 | |||
| 7 | #include <memory> | ||
| 8 | #include "core/frontend/input.h" | ||
| 9 | |||
| 10 | namespace InputCommon { | ||
| 11 | |||
| 12 | /** | ||
| 13 | * A touch device factory that takes a list of button devices and combines them into a touch device. | ||
| 14 | */ | ||
| 15 | class TouchFromButtonFactory final : public Input::Factory<Input::TouchDevice> { | ||
| 16 | public: | ||
| 17 | /** | ||
| 18 | * Creates a touch device from a list of button devices | ||
| 19 | */ | ||
| 20 | std::unique_ptr<Input::TouchDevice> Create(const Common::ParamPackage& params) override; | ||
| 21 | }; | ||
| 22 | |||
| 23 | } // namespace InputCommon | ||
diff --git a/src/video_core/macro/macro_jit_x64.cpp b/src/video_core/macro/macro_jit_x64.cpp index c1b9e4ad9..954b87515 100644 --- a/src/video_core/macro/macro_jit_x64.cpp +++ b/src/video_core/macro/macro_jit_x64.cpp | |||
| @@ -14,11 +14,11 @@ MICROPROFILE_DEFINE(MacroJitCompile, "GPU", "Compile macro JIT", MP_RGB(173, 255 | |||
| 14 | MICROPROFILE_DEFINE(MacroJitExecute, "GPU", "Execute macro JIT", MP_RGB(255, 255, 0)); | 14 | MICROPROFILE_DEFINE(MacroJitExecute, "GPU", "Execute macro JIT", MP_RGB(255, 255, 0)); |
| 15 | 15 | ||
| 16 | namespace Tegra { | 16 | namespace Tegra { |
| 17 | static const Xbyak::Reg64 STATE = Xbyak::util::rbx; | 17 | constexpr Xbyak::Reg64 STATE = Xbyak::util::rbx; |
| 18 | static const Xbyak::Reg32 RESULT = Xbyak::util::ebp; | 18 | constexpr Xbyak::Reg32 RESULT = Xbyak::util::ebp; |
| 19 | static const Xbyak::Reg64 PARAMETERS = Xbyak::util::r12; | 19 | constexpr Xbyak::Reg64 PARAMETERS = Xbyak::util::r12; |
| 20 | static const Xbyak::Reg32 METHOD_ADDRESS = Xbyak::util::r14d; | 20 | constexpr Xbyak::Reg32 METHOD_ADDRESS = Xbyak::util::r14d; |
| 21 | static const Xbyak::Reg64 BRANCH_HOLDER = Xbyak::util::r15; | 21 | constexpr Xbyak::Reg64 BRANCH_HOLDER = Xbyak::util::r15; |
| 22 | 22 | ||
| 23 | static const std::bitset<32> PERSISTENT_REGISTERS = Common::X64::BuildRegSet({ | 23 | static const std::bitset<32> PERSISTENT_REGISTERS = Common::X64::BuildRegSet({ |
| 24 | STATE, | 24 | STATE, |
diff --git a/src/video_core/renderer_vulkan/vk_device.cpp b/src/video_core/renderer_vulkan/vk_device.cpp index 030b4dbd3..4205bd573 100644 --- a/src/video_core/renderer_vulkan/vk_device.cpp +++ b/src/video_core/renderer_vulkan/vk_device.cpp | |||
| @@ -380,6 +380,14 @@ bool VKDevice::Create() { | |||
| 380 | 380 | ||
| 381 | CollectTelemetryParameters(); | 381 | CollectTelemetryParameters(); |
| 382 | 382 | ||
| 383 | if (ext_extended_dynamic_state && driver_id == VK_DRIVER_ID_AMD_PROPRIETARY_KHR) { | ||
| 384 | // AMD's proprietary driver supports VK_EXT_extended_dynamic_state but the <stride> field | ||
| 385 | // seems to be bugged. Blacklisting it for now. | ||
| 386 | LOG_WARNING(Render_Vulkan, | ||
| 387 | "Blacklisting AMD proprietary from VK_EXT_extended_dynamic_state"); | ||
| 388 | ext_extended_dynamic_state = false; | ||
| 389 | } | ||
| 390 | |||
| 383 | graphics_queue = logical.GetQueue(graphics_family); | 391 | graphics_queue = logical.GetQueue(graphics_family); |
| 384 | present_queue = logical.GetQueue(present_family); | 392 | present_queue = logical.GetQueue(present_family); |
| 385 | 393 | ||
| @@ -691,12 +699,7 @@ std::vector<const char*> VKDevice::LoadExtensions() { | |||
| 691 | } | 699 | } |
| 692 | } | 700 | } |
| 693 | 701 | ||
| 694 | if (has_ext_extended_dynamic_state && driver_id == VK_DRIVER_ID_AMD_PROPRIETARY) { | 702 | if (has_ext_extended_dynamic_state) { |
| 695 | // AMD's proprietary driver supports VK_EXT_extended_dynamic_state but the <stride> field | ||
| 696 | // seems to be bugged. Blacklisting it for now. | ||
| 697 | LOG_WARNING(Render_Vulkan, | ||
| 698 | "Blacklisting AMD proprietary from VK_EXT_extended_dynamic_state"); | ||
| 699 | } else if (has_ext_extended_dynamic_state) { | ||
| 700 | VkPhysicalDeviceExtendedDynamicStateFeaturesEXT dynamic_state; | 703 | VkPhysicalDeviceExtendedDynamicStateFeaturesEXT dynamic_state; |
| 701 | dynamic_state.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_EXTENDED_DYNAMIC_STATE_FEATURES_EXT; | 704 | dynamic_state.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_EXTENDED_DYNAMIC_STATE_FEATURES_EXT; |
| 702 | dynamic_state.pNext = nullptr; | 705 | dynamic_state.pNext = nullptr; |
diff --git a/src/video_core/shader/async_shaders.cpp b/src/video_core/shader/async_shaders.cpp index f815584f7..aabd62c5c 100644 --- a/src/video_core/shader/async_shaders.cpp +++ b/src/video_core/shader/async_shaders.cpp | |||
| @@ -73,11 +73,11 @@ void AsyncShaders::KillWorkers() { | |||
| 73 | worker_threads.clear(); | 73 | worker_threads.clear(); |
| 74 | } | 74 | } |
| 75 | 75 | ||
| 76 | bool AsyncShaders::HasWorkQueued() { | 76 | bool AsyncShaders::HasWorkQueued() const { |
| 77 | return !pending_queue.empty(); | 77 | return !pending_queue.empty(); |
| 78 | } | 78 | } |
| 79 | 79 | ||
| 80 | bool AsyncShaders::HasCompletedWork() { | 80 | bool AsyncShaders::HasCompletedWork() const { |
| 81 | std::shared_lock lock{completed_mutex}; | 81 | std::shared_lock lock{completed_mutex}; |
| 82 | return !finished_work.empty(); | 82 | return !finished_work.empty(); |
| 83 | } | 83 | } |
| @@ -102,7 +102,7 @@ bool AsyncShaders::IsShaderAsync(const Tegra::GPU& gpu) const { | |||
| 102 | } | 102 | } |
| 103 | 103 | ||
| 104 | std::vector<AsyncShaders::Result> AsyncShaders::GetCompletedWork() { | 104 | std::vector<AsyncShaders::Result> AsyncShaders::GetCompletedWork() { |
| 105 | std::vector<AsyncShaders::Result> results; | 105 | std::vector<Result> results; |
| 106 | { | 106 | { |
| 107 | std::unique_lock lock{completed_mutex}; | 107 | std::unique_lock lock{completed_mutex}; |
| 108 | results.assign(std::make_move_iterator(finished_work.begin()), | 108 | results.assign(std::make_move_iterator(finished_work.begin()), |
diff --git a/src/video_core/shader/async_shaders.h b/src/video_core/shader/async_shaders.h index d5ae814d5..7cf8d994c 100644 --- a/src/video_core/shader/async_shaders.h +++ b/src/video_core/shader/async_shaders.h | |||
| @@ -5,11 +5,10 @@ | |||
| 5 | #pragma once | 5 | #pragma once |
| 6 | 6 | ||
| 7 | #include <condition_variable> | 7 | #include <condition_variable> |
| 8 | #include <deque> | ||
| 9 | #include <memory> | 8 | #include <memory> |
| 10 | #include <shared_mutex> | 9 | #include <shared_mutex> |
| 11 | #include <thread> | 10 | #include <thread> |
| 12 | #include "common/bit_field.h" | 11 | |
| 13 | #include "common/common_types.h" | 12 | #include "common/common_types.h" |
| 14 | #include "video_core/renderer_opengl/gl_device.h" | 13 | #include "video_core/renderer_opengl/gl_device.h" |
| 15 | #include "video_core/renderer_opengl/gl_resource_manager.h" | 14 | #include "video_core/renderer_opengl/gl_resource_manager.h" |
| @@ -17,7 +16,6 @@ | |||
| 17 | #include "video_core/renderer_vulkan/vk_device.h" | 16 | #include "video_core/renderer_vulkan/vk_device.h" |
| 18 | #include "video_core/renderer_vulkan/vk_pipeline_cache.h" | 17 | #include "video_core/renderer_vulkan/vk_pipeline_cache.h" |
| 19 | #include "video_core/renderer_vulkan/vk_scheduler.h" | 18 | #include "video_core/renderer_vulkan/vk_scheduler.h" |
| 20 | #include "video_core/renderer_vulkan/vk_update_descriptor.h" | ||
| 21 | 19 | ||
| 22 | namespace Core::Frontend { | 20 | namespace Core::Frontend { |
| 23 | class EmuWindow; | 21 | class EmuWindow; |
| @@ -70,20 +68,20 @@ public: | |||
| 70 | void KillWorkers(); | 68 | void KillWorkers(); |
| 71 | 69 | ||
| 72 | /// Check to see if any shaders have actually been compiled | 70 | /// Check to see if any shaders have actually been compiled |
| 73 | bool HasCompletedWork(); | 71 | [[nodiscard]] bool HasCompletedWork() const; |
| 74 | 72 | ||
| 75 | /// Deduce if a shader can be build on another thread of MUST be built in sync. We cannot build | 73 | /// Deduce if a shader can be build on another thread of MUST be built in sync. We cannot build |
| 76 | /// every shader async as some shaders are only built and executed once. We try to "guess" which | 74 | /// every shader async as some shaders are only built and executed once. We try to "guess" which |
| 77 | /// shader would be used only once | 75 | /// shader would be used only once |
| 78 | bool IsShaderAsync(const Tegra::GPU& gpu) const; | 76 | [[nodiscard]] bool IsShaderAsync(const Tegra::GPU& gpu) const; |
| 79 | 77 | ||
| 80 | /// Pulls completed compiled shaders | 78 | /// Pulls completed compiled shaders |
| 81 | std::vector<Result> GetCompletedWork(); | 79 | [[nodiscard]] std::vector<Result> GetCompletedWork(); |
| 82 | 80 | ||
| 83 | void QueueOpenGLShader(const OpenGL::Device& device, Tegra::Engines::ShaderType shader_type, | 81 | void QueueOpenGLShader(const OpenGL::Device& device, Tegra::Engines::ShaderType shader_type, |
| 84 | u64 uid, std::vector<u64> code, std::vector<u64> code_b, u32 main_offset, | 82 | u64 uid, std::vector<u64> code, std::vector<u64> code_b, u32 main_offset, |
| 85 | VideoCommon::Shader::CompilerSettings compiler_settings, | 83 | CompilerSettings compiler_settings, const Registry& registry, |
| 86 | const VideoCommon::Shader::Registry& registry, VAddr cpu_addr); | 84 | VAddr cpu_addr); |
| 87 | 85 | ||
| 88 | void QueueVulkanShader(Vulkan::VKPipelineCache* pp_cache, const Vulkan::VKDevice& device, | 86 | void QueueVulkanShader(Vulkan::VKPipelineCache* pp_cache, const Vulkan::VKDevice& device, |
| 89 | Vulkan::VKScheduler& scheduler, | 87 | Vulkan::VKScheduler& scheduler, |
| @@ -97,7 +95,7 @@ private: | |||
| 97 | void ShaderCompilerThread(Core::Frontend::GraphicsContext* context); | 95 | void ShaderCompilerThread(Core::Frontend::GraphicsContext* context); |
| 98 | 96 | ||
| 99 | /// Check our worker queue to see if we have any work queued already | 97 | /// Check our worker queue to see if we have any work queued already |
| 100 | bool HasWorkQueued(); | 98 | [[nodiscard]] bool HasWorkQueued() const; |
| 101 | 99 | ||
| 102 | struct WorkerParams { | 100 | struct WorkerParams { |
| 103 | Backend backend; | 101 | Backend backend; |
| @@ -108,8 +106,8 @@ private: | |||
| 108 | std::vector<u64> code; | 106 | std::vector<u64> code; |
| 109 | std::vector<u64> code_b; | 107 | std::vector<u64> code_b; |
| 110 | u32 main_offset; | 108 | u32 main_offset; |
| 111 | VideoCommon::Shader::CompilerSettings compiler_settings; | 109 | CompilerSettings compiler_settings; |
| 112 | std::optional<VideoCommon::Shader::Registry> registry; | 110 | std::optional<Registry> registry; |
| 113 | VAddr cpu_address; | 111 | VAddr cpu_address; |
| 114 | 112 | ||
| 115 | // For Vulkan | 113 | // For Vulkan |
| @@ -125,13 +123,13 @@ private: | |||
| 125 | }; | 123 | }; |
| 126 | 124 | ||
| 127 | std::condition_variable cv; | 125 | std::condition_variable cv; |
| 128 | std::mutex queue_mutex; | 126 | mutable std::mutex queue_mutex; |
| 129 | std::shared_mutex completed_mutex; | 127 | mutable std::shared_mutex completed_mutex; |
| 130 | std::atomic<bool> is_thread_exiting{}; | 128 | std::atomic<bool> is_thread_exiting{}; |
| 131 | std::vector<std::unique_ptr<Core::Frontend::GraphicsContext>> context_list; | 129 | std::vector<std::unique_ptr<Core::Frontend::GraphicsContext>> context_list; |
| 132 | std::vector<std::thread> worker_threads; | 130 | std::vector<std::thread> worker_threads; |
| 133 | std::queue<WorkerParams> pending_queue; | 131 | std::queue<WorkerParams> pending_queue; |
| 134 | std::vector<AsyncShaders::Result> finished_work; | 132 | std::vector<Result> finished_work; |
| 135 | Core::Frontend::EmuWindow& emu_window; | 133 | Core::Frontend::EmuWindow& emu_window; |
| 136 | }; | 134 | }; |
| 137 | 135 | ||
diff --git a/src/yuzu/CMakeLists.txt b/src/yuzu/CMakeLists.txt index 6987e85e1..3ea4e5601 100644 --- a/src/yuzu/CMakeLists.txt +++ b/src/yuzu/CMakeLists.txt | |||
| @@ -68,6 +68,9 @@ add_executable(yuzu | |||
| 68 | configuration/configure_input_advanced.cpp | 68 | configuration/configure_input_advanced.cpp |
| 69 | configuration/configure_input_advanced.h | 69 | configuration/configure_input_advanced.h |
| 70 | configuration/configure_input_advanced.ui | 70 | configuration/configure_input_advanced.ui |
| 71 | configuration/configure_motion_touch.cpp | ||
| 72 | configuration/configure_motion_touch.h | ||
| 73 | configuration/configure_motion_touch.ui | ||
| 71 | configuration/configure_mouse_advanced.cpp | 74 | configuration/configure_mouse_advanced.cpp |
| 72 | configuration/configure_mouse_advanced.h | 75 | configuration/configure_mouse_advanced.h |
| 73 | configuration/configure_mouse_advanced.ui | 76 | configuration/configure_mouse_advanced.ui |
| @@ -86,9 +89,13 @@ add_executable(yuzu | |||
| 86 | configuration/configure_system.cpp | 89 | configuration/configure_system.cpp |
| 87 | configuration/configure_system.h | 90 | configuration/configure_system.h |
| 88 | configuration/configure_system.ui | 91 | configuration/configure_system.ui |
| 92 | configuration/configure_touch_from_button.cpp | ||
| 93 | configuration/configure_touch_from_button.h | ||
| 94 | configuration/configure_touch_from_button.ui | ||
| 89 | configuration/configure_touchscreen_advanced.cpp | 95 | configuration/configure_touchscreen_advanced.cpp |
| 90 | configuration/configure_touchscreen_advanced.h | 96 | configuration/configure_touchscreen_advanced.h |
| 91 | configuration/configure_touchscreen_advanced.ui | 97 | configuration/configure_touchscreen_advanced.ui |
| 98 | configuration/configure_touch_widget.h | ||
| 92 | configuration/configure_ui.cpp | 99 | configuration/configure_ui.cpp |
| 93 | configuration/configure_ui.h | 100 | configuration/configure_ui.h |
| 94 | configuration/configure_ui.ui | 101 | configuration/configure_ui.ui |
diff --git a/src/yuzu/bootmanager.cpp b/src/yuzu/bootmanager.cpp index f1b428bde..21707e451 100644 --- a/src/yuzu/bootmanager.cpp +++ b/src/yuzu/bootmanager.cpp | |||
| @@ -305,8 +305,8 @@ static Core::Frontend::EmuWindow::WindowSystemInfo GetWindowSystemInfo(QWindow* | |||
| 305 | } | 305 | } |
| 306 | 306 | ||
| 307 | GRenderWindow::GRenderWindow(GMainWindow* parent, EmuThread* emu_thread_, | 307 | GRenderWindow::GRenderWindow(GMainWindow* parent, EmuThread* emu_thread_, |
| 308 | InputCommon::InputSubsystem* input_subsystem_) | 308 | std::shared_ptr<InputCommon::InputSubsystem> input_subsystem_) |
| 309 | : QWidget(parent), emu_thread(emu_thread_), input_subsystem{input_subsystem_} { | 309 | : QWidget(parent), emu_thread(emu_thread_), input_subsystem{std::move(input_subsystem_)} { |
| 310 | setWindowTitle(QStringLiteral("yuzu %1 | %2-%3") | 310 | setWindowTitle(QStringLiteral("yuzu %1 | %2-%3") |
| 311 | .arg(QString::fromUtf8(Common::g_build_name), | 311 | .arg(QString::fromUtf8(Common::g_build_name), |
| 312 | QString::fromUtf8(Common::g_scm_branch), | 312 | QString::fromUtf8(Common::g_scm_branch), |
| @@ -452,7 +452,7 @@ void GRenderWindow::TouchUpdateEvent(const QTouchEvent* event) { | |||
| 452 | int active_points = 0; | 452 | int active_points = 0; |
| 453 | 453 | ||
| 454 | // average all active touch points | 454 | // average all active touch points |
| 455 | for (const auto tp : event->touchPoints()) { | 455 | for (const auto& tp : event->touchPoints()) { |
| 456 | if (tp.state() & (Qt::TouchPointPressed | Qt::TouchPointMoved | Qt::TouchPointStationary)) { | 456 | if (tp.state() & (Qt::TouchPointPressed | Qt::TouchPointMoved | Qt::TouchPointStationary)) { |
| 457 | active_points++; | 457 | active_points++; |
| 458 | pos += tp.pos(); | 458 | pos += tp.pos(); |
diff --git a/src/yuzu/bootmanager.h b/src/yuzu/bootmanager.h index ecb3b8135..ca35cf831 100644 --- a/src/yuzu/bootmanager.h +++ b/src/yuzu/bootmanager.h | |||
| @@ -6,6 +6,7 @@ | |||
| 6 | 6 | ||
| 7 | #include <atomic> | 7 | #include <atomic> |
| 8 | #include <condition_variable> | 8 | #include <condition_variable> |
| 9 | #include <memory> | ||
| 9 | #include <mutex> | 10 | #include <mutex> |
| 10 | 11 | ||
| 11 | #include <QImage> | 12 | #include <QImage> |
| @@ -126,7 +127,7 @@ class GRenderWindow : public QWidget, public Core::Frontend::EmuWindow { | |||
| 126 | 127 | ||
| 127 | public: | 128 | public: |
| 128 | explicit GRenderWindow(GMainWindow* parent, EmuThread* emu_thread_, | 129 | explicit GRenderWindow(GMainWindow* parent, EmuThread* emu_thread_, |
| 129 | InputCommon::InputSubsystem* input_subsystem_); | 130 | std::shared_ptr<InputCommon::InputSubsystem> input_subsystem_); |
| 130 | ~GRenderWindow() override; | 131 | ~GRenderWindow() override; |
| 131 | 132 | ||
| 132 | // EmuWindow implementation. | 133 | // EmuWindow implementation. |
| @@ -188,7 +189,7 @@ private: | |||
| 188 | QStringList GetUnsupportedGLExtensions() const; | 189 | QStringList GetUnsupportedGLExtensions() const; |
| 189 | 190 | ||
| 190 | EmuThread* emu_thread; | 191 | EmuThread* emu_thread; |
| 191 | InputCommon::InputSubsystem* input_subsystem; | 192 | std::shared_ptr<InputCommon::InputSubsystem> input_subsystem; |
| 192 | 193 | ||
| 193 | // Main context that will be shared with all other contexts that are requested. | 194 | // Main context that will be shared with all other contexts that are requested. |
| 194 | // If this is used in a shared context setting, then this should not be used directly, but | 195 | // If this is used in a shared context setting, then this should not be used directly, but |
diff --git a/src/yuzu/configuration/config.cpp b/src/yuzu/configuration/config.cpp index 489877be9..2bc55a26a 100644 --- a/src/yuzu/configuration/config.cpp +++ b/src/yuzu/configuration/config.cpp | |||
| @@ -51,8 +51,10 @@ const std::array<std::array<int, 4>, Settings::NativeAnalog::NumAnalogs> Config: | |||
| 51 | }, | 51 | }, |
| 52 | }}; | 52 | }}; |
| 53 | 53 | ||
| 54 | const int Config::default_lstick_mod = Qt::Key_E; | 54 | const std::array<int, 2> Config::default_stick_mod = { |
| 55 | const int Config::default_rstick_mod = Qt::Key_R; | 55 | Qt::Key_E, |
| 56 | Qt::Key_R, | ||
| 57 | }; | ||
| 56 | 58 | ||
| 57 | const std::array<int, Settings::NativeMouseButton::NumMouseButtons> Config::default_mouse_buttons = | 59 | const std::array<int, Settings::NativeMouseButton::NumMouseButtons> Config::default_mouse_buttons = |
| 58 | { | 60 | { |
| @@ -285,7 +287,7 @@ void Config::ReadPlayerValues() { | |||
| 285 | for (int i = 0; i < Settings::NativeAnalog::NumAnalogs; ++i) { | 287 | for (int i = 0; i < Settings::NativeAnalog::NumAnalogs; ++i) { |
| 286 | const std::string default_param = InputCommon::GenerateAnalogParamFromKeys( | 288 | const std::string default_param = InputCommon::GenerateAnalogParamFromKeys( |
| 287 | default_analogs[i][0], default_analogs[i][1], default_analogs[i][2], | 289 | default_analogs[i][0], default_analogs[i][1], default_analogs[i][2], |
| 288 | default_analogs[i][3], default_analogs[i][4], 0.5f); | 290 | default_analogs[i][3], default_stick_mod[i], 0.5f); |
| 289 | auto& player_analogs = player.analogs[i]; | 291 | auto& player_analogs = player.analogs[i]; |
| 290 | 292 | ||
| 291 | player_analogs = qt_config | 293 | player_analogs = qt_config |
| @@ -323,7 +325,7 @@ void Config::ReadDebugValues() { | |||
| 323 | for (int i = 0; i < Settings::NativeAnalog::NumAnalogs; ++i) { | 325 | for (int i = 0; i < Settings::NativeAnalog::NumAnalogs; ++i) { |
| 324 | const std::string default_param = InputCommon::GenerateAnalogParamFromKeys( | 326 | const std::string default_param = InputCommon::GenerateAnalogParamFromKeys( |
| 325 | default_analogs[i][0], default_analogs[i][1], default_analogs[i][2], | 327 | default_analogs[i][0], default_analogs[i][1], default_analogs[i][2], |
| 326 | default_analogs[i][3], default_analogs[i][4], 0.5f); | 328 | default_analogs[i][3], default_stick_mod[i], 0.5f); |
| 327 | auto& debug_pad_analogs = Settings::values.debug_pad_analogs[i]; | 329 | auto& debug_pad_analogs = Settings::values.debug_pad_analogs[i]; |
| 328 | 330 | ||
| 329 | debug_pad_analogs = qt_config | 331 | debug_pad_analogs = qt_config |
| @@ -418,14 +420,64 @@ void Config::ReadControlValues() { | |||
| 418 | ReadKeyboardValues(); | 420 | ReadKeyboardValues(); |
| 419 | ReadMouseValues(); | 421 | ReadMouseValues(); |
| 420 | ReadTouchscreenValues(); | 422 | ReadTouchscreenValues(); |
| 423 | ReadMotionTouchValues(); | ||
| 421 | 424 | ||
| 422 | Settings::values.vibration_enabled = | 425 | Settings::values.vibration_enabled = |
| 423 | ReadSetting(QStringLiteral("vibration_enabled"), true).toBool(); | 426 | ReadSetting(QStringLiteral("vibration_enabled"), true).toBool(); |
| 427 | Settings::values.use_docked_mode = | ||
| 428 | ReadSetting(QStringLiteral("use_docked_mode"), false).toBool(); | ||
| 429 | |||
| 430 | qt_config->endGroup(); | ||
| 431 | } | ||
| 432 | |||
| 433 | void Config::ReadMotionTouchValues() { | ||
| 434 | int num_touch_from_button_maps = | ||
| 435 | qt_config->beginReadArray(QStringLiteral("touch_from_button_maps")); | ||
| 436 | |||
| 437 | if (num_touch_from_button_maps > 0) { | ||
| 438 | const auto append_touch_from_button_map = [this] { | ||
| 439 | Settings::TouchFromButtonMap map; | ||
| 440 | map.name = ReadSetting(QStringLiteral("name"), QStringLiteral("default")) | ||
| 441 | .toString() | ||
| 442 | .toStdString(); | ||
| 443 | const int num_touch_maps = qt_config->beginReadArray(QStringLiteral("entries")); | ||
| 444 | map.buttons.reserve(num_touch_maps); | ||
| 445 | for (int i = 0; i < num_touch_maps; i++) { | ||
| 446 | qt_config->setArrayIndex(i); | ||
| 447 | std::string touch_mapping = | ||
| 448 | ReadSetting(QStringLiteral("bind")).toString().toStdString(); | ||
| 449 | map.buttons.emplace_back(std::move(touch_mapping)); | ||
| 450 | } | ||
| 451 | qt_config->endArray(); // entries | ||
| 452 | Settings::values.touch_from_button_maps.emplace_back(std::move(map)); | ||
| 453 | }; | ||
| 454 | |||
| 455 | for (int i = 0; i < num_touch_from_button_maps; ++i) { | ||
| 456 | qt_config->setArrayIndex(i); | ||
| 457 | append_touch_from_button_map(); | ||
| 458 | } | ||
| 459 | } else { | ||
| 460 | Settings::values.touch_from_button_maps.emplace_back( | ||
| 461 | Settings::TouchFromButtonMap{"default", {}}); | ||
| 462 | num_touch_from_button_maps = 1; | ||
| 463 | } | ||
| 464 | qt_config->endArray(); | ||
| 465 | |||
| 424 | Settings::values.motion_device = | 466 | Settings::values.motion_device = |
| 425 | ReadSetting(QStringLiteral("motion_device"), | 467 | ReadSetting(QStringLiteral("motion_device"), |
| 426 | QStringLiteral("engine:motion_emu,update_period:100,sensitivity:0.01")) | 468 | QStringLiteral("engine:motion_emu,update_period:100,sensitivity:0.01")) |
| 427 | .toString() | 469 | .toString() |
| 428 | .toStdString(); | 470 | .toStdString(); |
| 471 | Settings::values.touch_device = | ||
| 472 | ReadSetting(QStringLiteral("touch_device"), QStringLiteral("engine:emu_window")) | ||
| 473 | .toString() | ||
| 474 | .toStdString(); | ||
| 475 | Settings::values.use_touch_from_button = | ||
| 476 | ReadSetting(QStringLiteral("use_touch_from_button"), false).toBool(); | ||
| 477 | Settings::values.touch_from_button_map_index = | ||
| 478 | ReadSetting(QStringLiteral("touch_from_button_map"), 0).toInt(); | ||
| 479 | Settings::values.touch_from_button_map_index = | ||
| 480 | std::clamp(Settings::values.touch_from_button_map_index, 0, num_touch_from_button_maps - 1); | ||
| 429 | Settings::values.udp_input_address = | 481 | Settings::values.udp_input_address = |
| 430 | ReadSetting(QStringLiteral("udp_input_address"), | 482 | ReadSetting(QStringLiteral("udp_input_address"), |
| 431 | QString::fromUtf8(InputCommon::CemuhookUDP::DEFAULT_ADDR)) | 483 | QString::fromUtf8(InputCommon::CemuhookUDP::DEFAULT_ADDR)) |
| @@ -436,10 +488,6 @@ void Config::ReadControlValues() { | |||
| 436 | .toInt()); | 488 | .toInt()); |
| 437 | Settings::values.udp_pad_index = | 489 | Settings::values.udp_pad_index = |
| 438 | static_cast<u8>(ReadSetting(QStringLiteral("udp_pad_index"), 0).toUInt()); | 490 | static_cast<u8>(ReadSetting(QStringLiteral("udp_pad_index"), 0).toUInt()); |
| 439 | Settings::values.use_docked_mode = | ||
| 440 | ReadSetting(QStringLiteral("use_docked_mode"), false).toBool(); | ||
| 441 | |||
| 442 | qt_config->endGroup(); | ||
| 443 | } | 491 | } |
| 444 | 492 | ||
| 445 | void Config::ReadCoreValues() { | 493 | void Config::ReadCoreValues() { |
| @@ -877,7 +925,7 @@ void Config::SavePlayerValues() { | |||
| 877 | for (int i = 0; i < Settings::NativeAnalog::NumAnalogs; ++i) { | 925 | for (int i = 0; i < Settings::NativeAnalog::NumAnalogs; ++i) { |
| 878 | const std::string default_param = InputCommon::GenerateAnalogParamFromKeys( | 926 | const std::string default_param = InputCommon::GenerateAnalogParamFromKeys( |
| 879 | default_analogs[i][0], default_analogs[i][1], default_analogs[i][2], | 927 | default_analogs[i][0], default_analogs[i][1], default_analogs[i][2], |
| 880 | default_analogs[i][3], default_analogs[i][4], 0.5f); | 928 | default_analogs[i][3], default_stick_mod[i], 0.5f); |
| 881 | WriteSetting(QStringLiteral("player_%1_").arg(p) + | 929 | WriteSetting(QStringLiteral("player_%1_").arg(p) + |
| 882 | QString::fromStdString(Settings::NativeAnalog::mapping[i]), | 930 | QString::fromStdString(Settings::NativeAnalog::mapping[i]), |
| 883 | QString::fromStdString(player.analogs[i]), | 931 | QString::fromStdString(player.analogs[i]), |
| @@ -898,7 +946,7 @@ void Config::SaveDebugValues() { | |||
| 898 | for (int i = 0; i < Settings::NativeAnalog::NumAnalogs; ++i) { | 946 | for (int i = 0; i < Settings::NativeAnalog::NumAnalogs; ++i) { |
| 899 | const std::string default_param = InputCommon::GenerateAnalogParamFromKeys( | 947 | const std::string default_param = InputCommon::GenerateAnalogParamFromKeys( |
| 900 | default_analogs[i][0], default_analogs[i][1], default_analogs[i][2], | 948 | default_analogs[i][0], default_analogs[i][1], default_analogs[i][2], |
| 901 | default_analogs[i][3], default_analogs[i][4], 0.5f); | 949 | default_analogs[i][3], default_stick_mod[i], 0.5f); |
| 902 | WriteSetting(QStringLiteral("debug_pad_") + | 950 | WriteSetting(QStringLiteral("debug_pad_") + |
| 903 | QString::fromStdString(Settings::NativeAnalog::mapping[i]), | 951 | QString::fromStdString(Settings::NativeAnalog::mapping[i]), |
| 904 | QString::fromStdString(Settings::values.debug_pad_analogs[i]), | 952 | QString::fromStdString(Settings::values.debug_pad_analogs[i]), |
| @@ -932,6 +980,43 @@ void Config::SaveTouchscreenValues() { | |||
| 932 | WriteSetting(QStringLiteral("touchscreen_diameter_y"), touchscreen.diameter_y, 15); | 980 | WriteSetting(QStringLiteral("touchscreen_diameter_y"), touchscreen.diameter_y, 15); |
| 933 | } | 981 | } |
| 934 | 982 | ||
| 983 | void Config::SaveMotionTouchValues() { | ||
| 984 | WriteSetting(QStringLiteral("motion_device"), | ||
| 985 | QString::fromStdString(Settings::values.motion_device), | ||
| 986 | QStringLiteral("engine:motion_emu,update_period:100,sensitivity:0.01")); | ||
| 987 | WriteSetting(QStringLiteral("touch_device"), | ||
| 988 | QString::fromStdString(Settings::values.touch_device), | ||
| 989 | QStringLiteral("engine:emu_window")); | ||
| 990 | WriteSetting(QStringLiteral("use_touch_from_button"), Settings::values.use_touch_from_button, | ||
| 991 | false); | ||
| 992 | WriteSetting(QStringLiteral("touch_from_button_map"), | ||
| 993 | Settings::values.touch_from_button_map_index, 0); | ||
| 994 | WriteSetting(QStringLiteral("udp_input_address"), | ||
| 995 | QString::fromStdString(Settings::values.udp_input_address), | ||
| 996 | QString::fromUtf8(InputCommon::CemuhookUDP::DEFAULT_ADDR)); | ||
| 997 | WriteSetting(QStringLiteral("udp_input_port"), Settings::values.udp_input_port, | ||
| 998 | InputCommon::CemuhookUDP::DEFAULT_PORT); | ||
| 999 | WriteSetting(QStringLiteral("udp_pad_index"), Settings::values.udp_pad_index, 0); | ||
| 1000 | |||
| 1001 | qt_config->beginWriteArray(QStringLiteral("touch_from_button_maps")); | ||
| 1002 | for (std::size_t p = 0; p < Settings::values.touch_from_button_maps.size(); ++p) { | ||
| 1003 | qt_config->setArrayIndex(static_cast<int>(p)); | ||
| 1004 | WriteSetting(QStringLiteral("name"), | ||
| 1005 | QString::fromStdString(Settings::values.touch_from_button_maps[p].name), | ||
| 1006 | QStringLiteral("default")); | ||
| 1007 | qt_config->beginWriteArray(QStringLiteral("entries")); | ||
| 1008 | for (std::size_t q = 0; q < Settings::values.touch_from_button_maps[p].buttons.size(); | ||
| 1009 | ++q) { | ||
| 1010 | qt_config->setArrayIndex(static_cast<int>(q)); | ||
| 1011 | WriteSetting( | ||
| 1012 | QStringLiteral("bind"), | ||
| 1013 | QString::fromStdString(Settings::values.touch_from_button_maps[p].buttons[q])); | ||
| 1014 | } | ||
| 1015 | qt_config->endArray(); | ||
| 1016 | } | ||
| 1017 | qt_config->endArray(); | ||
| 1018 | } | ||
| 1019 | |||
| 935 | void Config::SaveValues() { | 1020 | void Config::SaveValues() { |
| 936 | if (global) { | 1021 | if (global) { |
| 937 | SaveControlValues(); | 1022 | SaveControlValues(); |
| @@ -974,18 +1059,16 @@ void Config::SaveControlValues() { | |||
| 974 | SaveDebugValues(); | 1059 | SaveDebugValues(); |
| 975 | SaveMouseValues(); | 1060 | SaveMouseValues(); |
| 976 | SaveTouchscreenValues(); | 1061 | SaveTouchscreenValues(); |
| 1062 | SaveMotionTouchValues(); | ||
| 977 | 1063 | ||
| 978 | WriteSetting(QStringLiteral("vibration_enabled"), Settings::values.vibration_enabled, true); | 1064 | WriteSetting(QStringLiteral("vibration_enabled"), Settings::values.vibration_enabled, true); |
| 979 | WriteSetting(QStringLiteral("motion_device"), | 1065 | WriteSetting(QStringLiteral("motion_device"), |
| 980 | QString::fromStdString(Settings::values.motion_device), | 1066 | QString::fromStdString(Settings::values.motion_device), |
| 981 | QStringLiteral("engine:motion_emu,update_period:100,sensitivity:0.01")); | 1067 | QStringLiteral("engine:motion_emu,update_period:100,sensitivity:0.01")); |
| 1068 | WriteSetting(QStringLiteral("touch_device"), | ||
| 1069 | QString::fromStdString(Settings::values.touch_device), | ||
| 1070 | QStringLiteral("engine:emu_window")); | ||
| 982 | WriteSetting(QStringLiteral("keyboard_enabled"), Settings::values.keyboard_enabled, false); | 1071 | WriteSetting(QStringLiteral("keyboard_enabled"), Settings::values.keyboard_enabled, false); |
| 983 | WriteSetting(QStringLiteral("udp_input_address"), | ||
| 984 | QString::fromStdString(Settings::values.udp_input_address), | ||
| 985 | QString::fromUtf8(InputCommon::CemuhookUDP::DEFAULT_ADDR)); | ||
| 986 | WriteSetting(QStringLiteral("udp_input_port"), Settings::values.udp_input_port, | ||
| 987 | InputCommon::CemuhookUDP::DEFAULT_PORT); | ||
| 988 | WriteSetting(QStringLiteral("udp_pad_index"), Settings::values.udp_pad_index, 0); | ||
| 989 | WriteSetting(QStringLiteral("use_docked_mode"), Settings::values.use_docked_mode, false); | 1072 | WriteSetting(QStringLiteral("use_docked_mode"), Settings::values.use_docked_mode, false); |
| 990 | 1073 | ||
| 991 | qt_config->endGroup(); | 1074 | qt_config->endGroup(); |
diff --git a/src/yuzu/configuration/config.h b/src/yuzu/configuration/config.h index 9eeaf9d1e..ca0d29c6c 100644 --- a/src/yuzu/configuration/config.h +++ b/src/yuzu/configuration/config.h | |||
| @@ -24,8 +24,7 @@ public: | |||
| 24 | 24 | ||
| 25 | static const std::array<int, Settings::NativeButton::NumButtons> default_buttons; | 25 | static const std::array<int, Settings::NativeButton::NumButtons> default_buttons; |
| 26 | static const std::array<std::array<int, 4>, Settings::NativeAnalog::NumAnalogs> default_analogs; | 26 | static const std::array<std::array<int, 4>, Settings::NativeAnalog::NumAnalogs> default_analogs; |
| 27 | static const int default_lstick_mod; | 27 | static const std::array<int, 2> default_stick_mod; |
| 28 | static const int default_rstick_mod; | ||
| 29 | static const std::array<int, Settings::NativeMouseButton::NumMouseButtons> | 28 | static const std::array<int, Settings::NativeMouseButton::NumMouseButtons> |
| 30 | default_mouse_buttons; | 29 | default_mouse_buttons; |
| 31 | static const std::array<int, Settings::NativeKeyboard::NumKeyboardKeys> default_keyboard_keys; | 30 | static const std::array<int, Settings::NativeKeyboard::NumKeyboardKeys> default_keyboard_keys; |
| @@ -39,6 +38,7 @@ private: | |||
| 39 | void ReadKeyboardValues(); | 38 | void ReadKeyboardValues(); |
| 40 | void ReadMouseValues(); | 39 | void ReadMouseValues(); |
| 41 | void ReadTouchscreenValues(); | 40 | void ReadTouchscreenValues(); |
| 41 | void ReadMotionTouchValues(); | ||
| 42 | 42 | ||
| 43 | // Read functions bases off the respective config section names. | 43 | // Read functions bases off the respective config section names. |
| 44 | void ReadAudioValues(); | 44 | void ReadAudioValues(); |
| @@ -65,6 +65,7 @@ private: | |||
| 65 | void SaveDebugValues(); | 65 | void SaveDebugValues(); |
| 66 | void SaveMouseValues(); | 66 | void SaveMouseValues(); |
| 67 | void SaveTouchscreenValues(); | 67 | void SaveTouchscreenValues(); |
| 68 | void SaveMotionTouchValues(); | ||
| 68 | 69 | ||
| 69 | // Save functions based off the respective config section names. | 70 | // Save functions based off the respective config section names. |
| 70 | void SaveAudioValues(); | 71 | void SaveAudioValues(); |
diff --git a/src/yuzu/configuration/configure_input.cpp b/src/yuzu/configuration/configure_input.cpp index 5223eed1d..ae3e31762 100644 --- a/src/yuzu/configuration/configure_input.cpp +++ b/src/yuzu/configuration/configure_input.cpp | |||
| @@ -20,6 +20,7 @@ | |||
| 20 | #include "yuzu/configuration/configure_input.h" | 20 | #include "yuzu/configuration/configure_input.h" |
| 21 | #include "yuzu/configuration/configure_input_advanced.h" | 21 | #include "yuzu/configuration/configure_input_advanced.h" |
| 22 | #include "yuzu/configuration/configure_input_player.h" | 22 | #include "yuzu/configuration/configure_input_player.h" |
| 23 | #include "yuzu/configuration/configure_motion_touch.h" | ||
| 23 | #include "yuzu/configuration/configure_mouse_advanced.h" | 24 | #include "yuzu/configuration/configure_mouse_advanced.h" |
| 24 | #include "yuzu/configuration/configure_touchscreen_advanced.h" | 25 | #include "yuzu/configuration/configure_touchscreen_advanced.h" |
| 25 | 26 | ||
| @@ -127,6 +128,10 @@ void ConfigureInput::Initialize(InputCommon::InputSubsystem* input_subsystem) { | |||
| 127 | }); | 128 | }); |
| 128 | connect(advanced, &ConfigureInputAdvanced::CallTouchscreenConfigDialog, | 129 | connect(advanced, &ConfigureInputAdvanced::CallTouchscreenConfigDialog, |
| 129 | [this] { CallConfigureDialog<ConfigureTouchscreenAdvanced>(*this); }); | 130 | [this] { CallConfigureDialog<ConfigureTouchscreenAdvanced>(*this); }); |
| 131 | connect(advanced, &ConfigureInputAdvanced::CallMotionTouchConfigDialog, | ||
| 132 | [this, input_subsystem] { | ||
| 133 | CallConfigureDialog<ConfigureMotionTouch>(*this, input_subsystem); | ||
| 134 | }); | ||
| 130 | 135 | ||
| 131 | connect(ui->buttonClearAll, &QPushButton::clicked, [this] { ClearAll(); }); | 136 | connect(ui->buttonClearAll, &QPushButton::clicked, [this] { ClearAll(); }); |
| 132 | connect(ui->buttonRestoreDefaults, &QPushButton::clicked, [this] { RestoreDefaults(); }); | 137 | connect(ui->buttonRestoreDefaults, &QPushButton::clicked, [this] { RestoreDefaults(); }); |
diff --git a/src/yuzu/configuration/configure_input_advanced.cpp b/src/yuzu/configuration/configure_input_advanced.cpp index db42b826b..81f9dc16c 100644 --- a/src/yuzu/configuration/configure_input_advanced.cpp +++ b/src/yuzu/configuration/configure_input_advanced.cpp | |||
| @@ -86,6 +86,8 @@ ConfigureInputAdvanced::ConfigureInputAdvanced(QWidget* parent) | |||
| 86 | connect(ui->mouse_advanced, &QPushButton::clicked, this, [this] { CallMouseConfigDialog(); }); | 86 | connect(ui->mouse_advanced, &QPushButton::clicked, this, [this] { CallMouseConfigDialog(); }); |
| 87 | connect(ui->touchscreen_advanced, &QPushButton::clicked, this, | 87 | connect(ui->touchscreen_advanced, &QPushButton::clicked, this, |
| 88 | [this] { CallTouchscreenConfigDialog(); }); | 88 | [this] { CallTouchscreenConfigDialog(); }); |
| 89 | connect(ui->buttonMotionTouch, &QPushButton::clicked, this, | ||
| 90 | &ConfigureInputAdvanced::CallMotionTouchConfigDialog); | ||
| 89 | 91 | ||
| 90 | LoadConfiguration(); | 92 | LoadConfiguration(); |
| 91 | } | 93 | } |
diff --git a/src/yuzu/configuration/configure_input_advanced.h b/src/yuzu/configuration/configure_input_advanced.h index d8fcec52d..50bb87768 100644 --- a/src/yuzu/configuration/configure_input_advanced.h +++ b/src/yuzu/configuration/configure_input_advanced.h | |||
| @@ -28,6 +28,7 @@ signals: | |||
| 28 | void CallDebugControllerDialog(); | 28 | void CallDebugControllerDialog(); |
| 29 | void CallMouseConfigDialog(); | 29 | void CallMouseConfigDialog(); |
| 30 | void CallTouchscreenConfigDialog(); | 30 | void CallTouchscreenConfigDialog(); |
| 31 | void CallMotionTouchConfigDialog(); | ||
| 31 | 32 | ||
| 32 | private: | 33 | private: |
| 33 | void changeEvent(QEvent* event) override; | 34 | void changeEvent(QEvent* event) override; |
diff --git a/src/yuzu/configuration/configure_input_player.cpp b/src/yuzu/configuration/configure_input_player.cpp index 80bf40acb..13ecb3dc5 100644 --- a/src/yuzu/configuration/configure_input_player.cpp +++ b/src/yuzu/configuration/configure_input_player.cpp | |||
| @@ -305,8 +305,8 @@ ConfigureInputPlayer::ConfigureInputPlayer(QWidget* parent, std::size_t player_i | |||
| 305 | } | 305 | } |
| 306 | 306 | ||
| 307 | // Handle clicks for the modifier buttons as well. | 307 | // Handle clicks for the modifier buttons as well. |
| 308 | ConfigureButtonClick(ui->buttonLStickMod, &lstick_mod, Config::default_lstick_mod); | 308 | ConfigureButtonClick(ui->buttonLStickMod, &lstick_mod, Config::default_stick_mod[0]); |
| 309 | ConfigureButtonClick(ui->buttonRStickMod, &rstick_mod, Config::default_rstick_mod); | 309 | ConfigureButtonClick(ui->buttonRStickMod, &rstick_mod, Config::default_stick_mod[1]); |
| 310 | 310 | ||
| 311 | for (int analog_id = 0; analog_id < Settings::NativeAnalog::NumAnalogs; ++analog_id) { | 311 | for (int analog_id = 0; analog_id < Settings::NativeAnalog::NumAnalogs; ++analog_id) { |
| 312 | for (int sub_button_id = 0; sub_button_id < ANALOG_SUB_BUTTONS_NUM; ++sub_button_id) { | 312 | for (int sub_button_id = 0; sub_button_id < ANALOG_SUB_BUTTONS_NUM; ++sub_button_id) { |
| @@ -532,9 +532,9 @@ void ConfigureInputPlayer::RestoreDefaults() { | |||
| 532 | 532 | ||
| 533 | // Reset Modifier Buttons | 533 | // Reset Modifier Buttons |
| 534 | lstick_mod = | 534 | lstick_mod = |
| 535 | Common::ParamPackage(InputCommon::GenerateKeyboardParam(Config::default_lstick_mod)); | 535 | Common::ParamPackage(InputCommon::GenerateKeyboardParam(Config::default_stick_mod[0])); |
| 536 | rstick_mod = | 536 | rstick_mod = |
| 537 | Common::ParamPackage(InputCommon::GenerateKeyboardParam(Config::default_rstick_mod)); | 537 | Common::ParamPackage(InputCommon::GenerateKeyboardParam(Config::default_stick_mod[1])); |
| 538 | 538 | ||
| 539 | // Reset Analogs | 539 | // Reset Analogs |
| 540 | for (int analog_id = 0; analog_id < Settings::NativeAnalog::NumAnalogs; ++analog_id) { | 540 | for (int analog_id = 0; analog_id < Settings::NativeAnalog::NumAnalogs; ++analog_id) { |
diff --git a/src/yuzu/configuration/configure_motion_touch.cpp b/src/yuzu/configuration/configure_motion_touch.cpp new file mode 100644 index 000000000..c7d085151 --- /dev/null +++ b/src/yuzu/configuration/configure_motion_touch.cpp | |||
| @@ -0,0 +1,314 @@ | |||
| 1 | // Copyright 2018 Citra Emulator Project | ||
| 2 | // Licensed under GPLv2 or any later version | ||
| 3 | // Refer to the license.txt file included. | ||
| 4 | |||
| 5 | #include <array> | ||
| 6 | #include <QCloseEvent> | ||
| 7 | #include <QLabel> | ||
| 8 | #include <QMessageBox> | ||
| 9 | #include <QPushButton> | ||
| 10 | #include <QVBoxLayout> | ||
| 11 | #include "common/logging/log.h" | ||
| 12 | #include "core/settings.h" | ||
| 13 | #include "input_common/main.h" | ||
| 14 | #include "input_common/udp/client.h" | ||
| 15 | #include "input_common/udp/udp.h" | ||
| 16 | #include "ui_configure_motion_touch.h" | ||
| 17 | #include "yuzu/configuration/configure_motion_touch.h" | ||
| 18 | #include "yuzu/configuration/configure_touch_from_button.h" | ||
| 19 | |||
| 20 | CalibrationConfigurationDialog::CalibrationConfigurationDialog(QWidget* parent, | ||
| 21 | const std::string& host, u16 port, | ||
| 22 | u8 pad_index, u16 client_id) | ||
| 23 | : QDialog(parent) { | ||
| 24 | layout = new QVBoxLayout; | ||
| 25 | status_label = new QLabel(tr("Communicating with the server...")); | ||
| 26 | cancel_button = new QPushButton(tr("Cancel")); | ||
| 27 | connect(cancel_button, &QPushButton::clicked, this, [this] { | ||
| 28 | if (!completed) { | ||
| 29 | job->Stop(); | ||
| 30 | } | ||
| 31 | accept(); | ||
| 32 | }); | ||
| 33 | layout->addWidget(status_label); | ||
| 34 | layout->addWidget(cancel_button); | ||
| 35 | setLayout(layout); | ||
| 36 | |||
| 37 | using namespace InputCommon::CemuhookUDP; | ||
| 38 | job = std::make_unique<CalibrationConfigurationJob>( | ||
| 39 | host, port, pad_index, client_id, | ||
| 40 | [this](CalibrationConfigurationJob::Status status) { | ||
| 41 | QString text; | ||
| 42 | switch (status) { | ||
| 43 | case CalibrationConfigurationJob::Status::Ready: | ||
| 44 | text = tr("Touch the top left corner <br>of your touchpad."); | ||
| 45 | break; | ||
| 46 | case CalibrationConfigurationJob::Status::Stage1Completed: | ||
| 47 | text = tr("Now touch the bottom right corner <br>of your touchpad."); | ||
| 48 | break; | ||
| 49 | case CalibrationConfigurationJob::Status::Completed: | ||
| 50 | text = tr("Configuration completed!"); | ||
| 51 | break; | ||
| 52 | } | ||
| 53 | QMetaObject::invokeMethod(this, "UpdateLabelText", Q_ARG(QString, text)); | ||
| 54 | if (status == CalibrationConfigurationJob::Status::Completed) { | ||
| 55 | QMetaObject::invokeMethod(this, "UpdateButtonText", Q_ARG(QString, tr("OK"))); | ||
| 56 | } | ||
| 57 | }, | ||
| 58 | [this](u16 min_x_, u16 min_y_, u16 max_x_, u16 max_y_) { | ||
| 59 | completed = true; | ||
| 60 | min_x = min_x_; | ||
| 61 | min_y = min_y_; | ||
| 62 | max_x = max_x_; | ||
| 63 | max_y = max_y_; | ||
| 64 | }); | ||
| 65 | } | ||
| 66 | |||
| 67 | CalibrationConfigurationDialog::~CalibrationConfigurationDialog() = default; | ||
| 68 | |||
| 69 | void CalibrationConfigurationDialog::UpdateLabelText(const QString& text) { | ||
| 70 | status_label->setText(text); | ||
| 71 | } | ||
| 72 | |||
| 73 | void CalibrationConfigurationDialog::UpdateButtonText(const QString& text) { | ||
| 74 | cancel_button->setText(text); | ||
| 75 | } | ||
| 76 | |||
| 77 | constexpr std::array<std::pair<const char*, const char*>, 2> MotionProviders = {{ | ||
| 78 | {"motion_emu", QT_TRANSLATE_NOOP("ConfigureMotionTouch", "Mouse (Right Click)")}, | ||
| 79 | {"cemuhookudp", QT_TRANSLATE_NOOP("ConfigureMotionTouch", "CemuhookUDP")}, | ||
| 80 | }}; | ||
| 81 | |||
| 82 | constexpr std::array<std::pair<const char*, const char*>, 2> TouchProviders = {{ | ||
| 83 | {"emu_window", QT_TRANSLATE_NOOP("ConfigureMotionTouch", "Emulator Window")}, | ||
| 84 | {"cemuhookudp", QT_TRANSLATE_NOOP("ConfigureMotionTouch", "CemuhookUDP")}, | ||
| 85 | }}; | ||
| 86 | |||
| 87 | ConfigureMotionTouch::ConfigureMotionTouch(QWidget* parent, | ||
| 88 | InputCommon::InputSubsystem* input_subsystem_) | ||
| 89 | : QDialog(parent), input_subsystem{input_subsystem_}, | ||
| 90 | ui(std::make_unique<Ui::ConfigureMotionTouch>()) { | ||
| 91 | ui->setupUi(this); | ||
| 92 | for (const auto& [provider, name] : MotionProviders) { | ||
| 93 | ui->motion_provider->addItem(tr(name), QString::fromUtf8(provider)); | ||
| 94 | } | ||
| 95 | for (const auto& [provider, name] : TouchProviders) { | ||
| 96 | ui->touch_provider->addItem(tr(name), QString::fromUtf8(provider)); | ||
| 97 | } | ||
| 98 | |||
| 99 | ui->udp_learn_more->setOpenExternalLinks(true); | ||
| 100 | ui->udp_learn_more->setText( | ||
| 101 | tr("<a " | ||
| 102 | "href='https://yuzu-emu.org/wiki/" | ||
| 103 | "using-a-controller-or-android-phone-for-motion-or-touch-input'><span " | ||
| 104 | "style=\"text-decoration: underline; color:#039be5;\">Learn More</span></a>")); | ||
| 105 | |||
| 106 | SetConfiguration(); | ||
| 107 | UpdateUiDisplay(); | ||
| 108 | ConnectEvents(); | ||
| 109 | } | ||
| 110 | |||
| 111 | ConfigureMotionTouch::~ConfigureMotionTouch() = default; | ||
| 112 | |||
| 113 | void ConfigureMotionTouch::SetConfiguration() { | ||
| 114 | const Common::ParamPackage motion_param(Settings::values.motion_device); | ||
| 115 | const Common::ParamPackage touch_param(Settings::values.touch_device); | ||
| 116 | const std::string motion_engine = motion_param.Get("engine", "motion_emu"); | ||
| 117 | const std::string touch_engine = touch_param.Get("engine", "emu_window"); | ||
| 118 | |||
| 119 | ui->motion_provider->setCurrentIndex( | ||
| 120 | ui->motion_provider->findData(QString::fromStdString(motion_engine))); | ||
| 121 | ui->touch_provider->setCurrentIndex( | ||
| 122 | ui->touch_provider->findData(QString::fromStdString(touch_engine))); | ||
| 123 | ui->touch_from_button_checkbox->setChecked(Settings::values.use_touch_from_button); | ||
| 124 | touch_from_button_maps = Settings::values.touch_from_button_maps; | ||
| 125 | for (const auto& touch_map : touch_from_button_maps) { | ||
| 126 | ui->touch_from_button_map->addItem(QString::fromStdString(touch_map.name)); | ||
| 127 | } | ||
| 128 | ui->touch_from_button_map->setCurrentIndex(Settings::values.touch_from_button_map_index); | ||
| 129 | ui->motion_sensitivity->setValue(motion_param.Get("sensitivity", 0.01f)); | ||
| 130 | |||
| 131 | min_x = touch_param.Get("min_x", 100); | ||
| 132 | min_y = touch_param.Get("min_y", 50); | ||
| 133 | max_x = touch_param.Get("max_x", 1800); | ||
| 134 | max_y = touch_param.Get("max_y", 850); | ||
| 135 | |||
| 136 | ui->udp_server->setText(QString::fromStdString(Settings::values.udp_input_address)); | ||
| 137 | ui->udp_port->setText(QString::number(Settings::values.udp_input_port)); | ||
| 138 | ui->udp_pad_index->setCurrentIndex(Settings::values.udp_pad_index); | ||
| 139 | } | ||
| 140 | |||
| 141 | void ConfigureMotionTouch::UpdateUiDisplay() { | ||
| 142 | const QString motion_engine = ui->motion_provider->currentData().toString(); | ||
| 143 | const QString touch_engine = ui->touch_provider->currentData().toString(); | ||
| 144 | const QString cemuhook_udp = QStringLiteral("cemuhookudp"); | ||
| 145 | |||
| 146 | if (motion_engine == QStringLiteral("motion_emu")) { | ||
| 147 | ui->motion_sensitivity_label->setVisible(true); | ||
| 148 | ui->motion_sensitivity->setVisible(true); | ||
| 149 | } else { | ||
| 150 | ui->motion_sensitivity_label->setVisible(false); | ||
| 151 | ui->motion_sensitivity->setVisible(false); | ||
| 152 | } | ||
| 153 | |||
| 154 | if (touch_engine == cemuhook_udp) { | ||
| 155 | ui->touch_calibration->setVisible(true); | ||
| 156 | ui->touch_calibration_config->setVisible(true); | ||
| 157 | ui->touch_calibration_label->setVisible(true); | ||
| 158 | ui->touch_calibration->setText( | ||
| 159 | QStringLiteral("(%1, %2) - (%3, %4)").arg(min_x).arg(min_y).arg(max_x).arg(max_y)); | ||
| 160 | } else { | ||
| 161 | ui->touch_calibration->setVisible(false); | ||
| 162 | ui->touch_calibration_config->setVisible(false); | ||
| 163 | ui->touch_calibration_label->setVisible(false); | ||
| 164 | } | ||
| 165 | |||
| 166 | if (motion_engine == cemuhook_udp || touch_engine == cemuhook_udp) { | ||
| 167 | ui->udp_config_group_box->setVisible(true); | ||
| 168 | } else { | ||
| 169 | ui->udp_config_group_box->setVisible(false); | ||
| 170 | } | ||
| 171 | } | ||
| 172 | |||
| 173 | void ConfigureMotionTouch::ConnectEvents() { | ||
| 174 | connect(ui->motion_provider, qOverload<int>(&QComboBox::currentIndexChanged), this, | ||
| 175 | [this](int index) { UpdateUiDisplay(); }); | ||
| 176 | connect(ui->touch_provider, qOverload<int>(&QComboBox::currentIndexChanged), this, | ||
| 177 | [this](int index) { UpdateUiDisplay(); }); | ||
| 178 | connect(ui->udp_test, &QPushButton::clicked, this, &ConfigureMotionTouch::OnCemuhookUDPTest); | ||
| 179 | connect(ui->touch_calibration_config, &QPushButton::clicked, this, | ||
| 180 | &ConfigureMotionTouch::OnConfigureTouchCalibration); | ||
| 181 | connect(ui->touch_from_button_config_btn, &QPushButton::clicked, this, | ||
| 182 | &ConfigureMotionTouch::OnConfigureTouchFromButton); | ||
| 183 | connect(ui->buttonBox, &QDialogButtonBox::rejected, this, [this] { | ||
| 184 | if (CanCloseDialog()) { | ||
| 185 | reject(); | ||
| 186 | } | ||
| 187 | }); | ||
| 188 | } | ||
| 189 | |||
| 190 | void ConfigureMotionTouch::OnCemuhookUDPTest() { | ||
| 191 | ui->udp_test->setEnabled(false); | ||
| 192 | ui->udp_test->setText(tr("Testing")); | ||
| 193 | udp_test_in_progress = true; | ||
| 194 | InputCommon::CemuhookUDP::TestCommunication( | ||
| 195 | ui->udp_server->text().toStdString(), static_cast<u16>(ui->udp_port->text().toInt()), | ||
| 196 | static_cast<u8>(ui->udp_pad_index->currentIndex()), 24872, | ||
| 197 | [this] { | ||
| 198 | LOG_INFO(Frontend, "UDP input test success"); | ||
| 199 | QMetaObject::invokeMethod(this, "ShowUDPTestResult", Q_ARG(bool, true)); | ||
| 200 | }, | ||
| 201 | [this] { | ||
| 202 | LOG_ERROR(Frontend, "UDP input test failed"); | ||
| 203 | QMetaObject::invokeMethod(this, "ShowUDPTestResult", Q_ARG(bool, false)); | ||
| 204 | }); | ||
| 205 | } | ||
| 206 | |||
| 207 | void ConfigureMotionTouch::OnConfigureTouchCalibration() { | ||
| 208 | ui->touch_calibration_config->setEnabled(false); | ||
| 209 | ui->touch_calibration_config->setText(tr("Configuring")); | ||
| 210 | CalibrationConfigurationDialog dialog( | ||
| 211 | this, ui->udp_server->text().toStdString(), static_cast<u16>(ui->udp_port->text().toUInt()), | ||
| 212 | static_cast<u8>(ui->udp_pad_index->currentIndex()), 24872); | ||
| 213 | dialog.exec(); | ||
| 214 | if (dialog.completed) { | ||
| 215 | min_x = dialog.min_x; | ||
| 216 | min_y = dialog.min_y; | ||
| 217 | max_x = dialog.max_x; | ||
| 218 | max_y = dialog.max_y; | ||
| 219 | LOG_INFO(Frontend, | ||
| 220 | "UDP touchpad calibration config success: min_x={}, min_y={}, max_x={}, max_y={}", | ||
| 221 | min_x, min_y, max_x, max_y); | ||
| 222 | UpdateUiDisplay(); | ||
| 223 | } else { | ||
| 224 | LOG_ERROR(Frontend, "UDP touchpad calibration config failed"); | ||
| 225 | } | ||
| 226 | ui->touch_calibration_config->setEnabled(true); | ||
| 227 | ui->touch_calibration_config->setText(tr("Configure")); | ||
| 228 | } | ||
| 229 | |||
| 230 | void ConfigureMotionTouch::closeEvent(QCloseEvent* event) { | ||
| 231 | if (CanCloseDialog()) { | ||
| 232 | event->accept(); | ||
| 233 | } else { | ||
| 234 | event->ignore(); | ||
| 235 | } | ||
| 236 | } | ||
| 237 | |||
| 238 | void ConfigureMotionTouch::ShowUDPTestResult(bool result) { | ||
| 239 | udp_test_in_progress = false; | ||
| 240 | if (result) { | ||
| 241 | QMessageBox::information(this, tr("Test Successful"), | ||
| 242 | tr("Successfully received data from the server.")); | ||
| 243 | } else { | ||
| 244 | QMessageBox::warning(this, tr("Test Failed"), | ||
| 245 | tr("Could not receive valid data from the server.<br>Please verify " | ||
| 246 | "that the server is set up correctly and " | ||
| 247 | "the address and port are correct.")); | ||
| 248 | } | ||
| 249 | ui->udp_test->setEnabled(true); | ||
| 250 | ui->udp_test->setText(tr("Test")); | ||
| 251 | } | ||
| 252 | |||
| 253 | void ConfigureMotionTouch::OnConfigureTouchFromButton() { | ||
| 254 | ConfigureTouchFromButton dialog{this, touch_from_button_maps, input_subsystem, | ||
| 255 | ui->touch_from_button_map->currentIndex()}; | ||
| 256 | if (dialog.exec() != QDialog::Accepted) { | ||
| 257 | return; | ||
| 258 | } | ||
| 259 | touch_from_button_maps = dialog.GetMaps(); | ||
| 260 | |||
| 261 | while (ui->touch_from_button_map->count() > 0) { | ||
| 262 | ui->touch_from_button_map->removeItem(0); | ||
| 263 | } | ||
| 264 | for (const auto& touch_map : touch_from_button_maps) { | ||
| 265 | ui->touch_from_button_map->addItem(QString::fromStdString(touch_map.name)); | ||
| 266 | } | ||
| 267 | ui->touch_from_button_map->setCurrentIndex(dialog.GetSelectedIndex()); | ||
| 268 | } | ||
| 269 | |||
| 270 | bool ConfigureMotionTouch::CanCloseDialog() { | ||
| 271 | if (udp_test_in_progress) { | ||
| 272 | QMessageBox::warning(this, tr("Citra"), | ||
| 273 | tr("UDP Test or calibration configuration is in progress.<br>Please " | ||
| 274 | "wait for them to finish.")); | ||
| 275 | return false; | ||
| 276 | } | ||
| 277 | return true; | ||
| 278 | } | ||
| 279 | |||
| 280 | void ConfigureMotionTouch::ApplyConfiguration() { | ||
| 281 | if (!CanCloseDialog()) { | ||
| 282 | return; | ||
| 283 | } | ||
| 284 | |||
| 285 | std::string motion_engine = ui->motion_provider->currentData().toString().toStdString(); | ||
| 286 | std::string touch_engine = ui->touch_provider->currentData().toString().toStdString(); | ||
| 287 | |||
| 288 | Common::ParamPackage motion_param{}, touch_param{}; | ||
| 289 | motion_param.Set("engine", std::move(motion_engine)); | ||
| 290 | touch_param.Set("engine", std::move(touch_engine)); | ||
| 291 | |||
| 292 | if (motion_engine == "motion_emu") { | ||
| 293 | motion_param.Set("sensitivity", static_cast<float>(ui->motion_sensitivity->value())); | ||
| 294 | } | ||
| 295 | |||
| 296 | if (touch_engine == "cemuhookudp") { | ||
| 297 | touch_param.Set("min_x", min_x); | ||
| 298 | touch_param.Set("min_y", min_y); | ||
| 299 | touch_param.Set("max_x", max_x); | ||
| 300 | touch_param.Set("max_y", max_y); | ||
| 301 | } | ||
| 302 | |||
| 303 | Settings::values.motion_device = motion_param.Serialize(); | ||
| 304 | Settings::values.touch_device = touch_param.Serialize(); | ||
| 305 | Settings::values.use_touch_from_button = ui->touch_from_button_checkbox->isChecked(); | ||
| 306 | Settings::values.touch_from_button_map_index = ui->touch_from_button_map->currentIndex(); | ||
| 307 | Settings::values.touch_from_button_maps = touch_from_button_maps; | ||
| 308 | Settings::values.udp_input_address = ui->udp_server->text().toStdString(); | ||
| 309 | Settings::values.udp_input_port = static_cast<u16>(ui->udp_port->text().toInt()); | ||
| 310 | Settings::values.udp_pad_index = static_cast<u8>(ui->udp_pad_index->currentIndex()); | ||
| 311 | input_subsystem->ReloadInputDevices(); | ||
| 312 | |||
| 313 | accept(); | ||
| 314 | } | ||
diff --git a/src/yuzu/configuration/configure_motion_touch.h b/src/yuzu/configuration/configure_motion_touch.h new file mode 100644 index 000000000..3d4b5d659 --- /dev/null +++ b/src/yuzu/configuration/configure_motion_touch.h | |||
| @@ -0,0 +1,90 @@ | |||
| 1 | // Copyright 2018 Citra Emulator Project | ||
| 2 | // Licensed under GPLv2 or any later version | ||
| 3 | // Refer to the license.txt file included. | ||
| 4 | |||
| 5 | #pragma once | ||
| 6 | |||
| 7 | #include <memory> | ||
| 8 | #include <QDialog> | ||
| 9 | #include "common/param_package.h" | ||
| 10 | |||
| 11 | class QLabel; | ||
| 12 | class QPushButton; | ||
| 13 | class QVBoxLayout; | ||
| 14 | |||
| 15 | namespace InputCommon { | ||
| 16 | class InputSubsystem; | ||
| 17 | } | ||
| 18 | |||
| 19 | namespace InputCommon::CemuhookUDP { | ||
| 20 | class CalibrationConfigurationJob; | ||
| 21 | } | ||
| 22 | |||
| 23 | namespace Ui { | ||
| 24 | class ConfigureMotionTouch; | ||
| 25 | } | ||
| 26 | |||
| 27 | /// A dialog for touchpad calibration configuration. | ||
| 28 | class CalibrationConfigurationDialog : public QDialog { | ||
| 29 | Q_OBJECT | ||
| 30 | public: | ||
| 31 | explicit CalibrationConfigurationDialog(QWidget* parent, const std::string& host, u16 port, | ||
| 32 | u8 pad_index, u16 client_id); | ||
| 33 | ~CalibrationConfigurationDialog() override; | ||
| 34 | |||
| 35 | private: | ||
| 36 | Q_INVOKABLE void UpdateLabelText(const QString& text); | ||
| 37 | Q_INVOKABLE void UpdateButtonText(const QString& text); | ||
| 38 | |||
| 39 | QVBoxLayout* layout; | ||
| 40 | QLabel* status_label; | ||
| 41 | QPushButton* cancel_button; | ||
| 42 | std::unique_ptr<InputCommon::CemuhookUDP::CalibrationConfigurationJob> job; | ||
| 43 | |||
| 44 | // Configuration results | ||
| 45 | bool completed{}; | ||
| 46 | u16 min_x{}; | ||
| 47 | u16 min_y{}; | ||
| 48 | u16 max_x{}; | ||
| 49 | u16 max_y{}; | ||
| 50 | |||
| 51 | friend class ConfigureMotionTouch; | ||
| 52 | }; | ||
| 53 | |||
| 54 | class ConfigureMotionTouch : public QDialog { | ||
| 55 | Q_OBJECT | ||
| 56 | |||
| 57 | public: | ||
| 58 | explicit ConfigureMotionTouch(QWidget* parent, InputCommon::InputSubsystem* input_subsystem_); | ||
| 59 | ~ConfigureMotionTouch() override; | ||
| 60 | |||
| 61 | public slots: | ||
| 62 | void ApplyConfiguration(); | ||
| 63 | |||
| 64 | private slots: | ||
| 65 | void OnCemuhookUDPTest(); | ||
| 66 | void OnConfigureTouchCalibration(); | ||
| 67 | void OnConfigureTouchFromButton(); | ||
| 68 | |||
| 69 | private: | ||
| 70 | void closeEvent(QCloseEvent* event) override; | ||
| 71 | Q_INVOKABLE void ShowUDPTestResult(bool result); | ||
| 72 | void SetConfiguration(); | ||
| 73 | void UpdateUiDisplay(); | ||
| 74 | void ConnectEvents(); | ||
| 75 | bool CanCloseDialog(); | ||
| 76 | |||
| 77 | InputCommon::InputSubsystem* input_subsystem; | ||
| 78 | |||
| 79 | std::unique_ptr<Ui::ConfigureMotionTouch> ui; | ||
| 80 | |||
| 81 | // Coordinate system of the CemuhookUDP touch provider | ||
| 82 | int min_x{}; | ||
| 83 | int min_y{}; | ||
| 84 | int max_x{}; | ||
| 85 | int max_y{}; | ||
| 86 | |||
| 87 | bool udp_test_in_progress{}; | ||
| 88 | |||
| 89 | std::vector<Settings::TouchFromButtonMap> touch_from_button_maps; | ||
| 90 | }; | ||
diff --git a/src/yuzu/configuration/configure_motion_touch.ui b/src/yuzu/configuration/configure_motion_touch.ui new file mode 100644 index 000000000..602cf8cd8 --- /dev/null +++ b/src/yuzu/configuration/configure_motion_touch.ui | |||
| @@ -0,0 +1,327 @@ | |||
| 1 | <?xml version="1.0" encoding="UTF-8"?> | ||
| 2 | <ui version="4.0"> | ||
| 3 | <class>ConfigureMotionTouch</class> | ||
| 4 | <widget class="QDialog" name="ConfigureMotionTouch"> | ||
| 5 | <property name="windowTitle"> | ||
| 6 | <string>Configure Motion / Touch</string> | ||
| 7 | </property> | ||
| 8 | <property name="geometry"> | ||
| 9 | <rect> | ||
| 10 | <x>0</x> | ||
| 11 | <y>0</y> | ||
| 12 | <width>500</width> | ||
| 13 | <height>450</height> | ||
| 14 | </rect> | ||
| 15 | </property> | ||
| 16 | <layout class="QVBoxLayout"> | ||
| 17 | <item> | ||
| 18 | <widget class="QGroupBox" name="motion_group_box"> | ||
| 19 | <property name="title"> | ||
| 20 | <string>Motion</string> | ||
| 21 | </property> | ||
| 22 | <layout class="QVBoxLayout"> | ||
| 23 | <item> | ||
| 24 | <layout class="QHBoxLayout"> | ||
| 25 | <item> | ||
| 26 | <widget class="QLabel" name="motion_provider_label"> | ||
| 27 | <property name="text"> | ||
| 28 | <string>Motion Provider:</string> | ||
| 29 | </property> | ||
| 30 | </widget> | ||
| 31 | </item> | ||
| 32 | <item> | ||
| 33 | <widget class="QComboBox" name="motion_provider"/> | ||
| 34 | </item> | ||
| 35 | </layout> | ||
| 36 | </item> | ||
| 37 | <item> | ||
| 38 | <layout class="QHBoxLayout"> | ||
| 39 | <item> | ||
| 40 | <widget class="QLabel" name="motion_sensitivity_label"> | ||
| 41 | <property name="text"> | ||
| 42 | <string>Sensitivity:</string> | ||
| 43 | </property> | ||
| 44 | </widget> | ||
| 45 | </item> | ||
| 46 | <item> | ||
| 47 | <widget class="QDoubleSpinBox" name="motion_sensitivity"> | ||
| 48 | <property name="alignment"> | ||
| 49 | <set>Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter</set> | ||
| 50 | </property> | ||
| 51 | <property name="decimals"> | ||
| 52 | <number>4</number> | ||
| 53 | </property> | ||
| 54 | <property name="minimum"> | ||
| 55 | <double>0.010000000000000</double> | ||
| 56 | </property> | ||
| 57 | <property name="maximum"> | ||
| 58 | <double>10.000000000000000</double> | ||
| 59 | </property> | ||
| 60 | <property name="singleStep"> | ||
| 61 | <double>0.001000000000000</double> | ||
| 62 | </property> | ||
| 63 | <property name="value"> | ||
| 64 | <double>0.010000000000000</double> | ||
| 65 | </property> | ||
| 66 | </widget> | ||
| 67 | </item> | ||
| 68 | </layout> | ||
| 69 | </item> | ||
| 70 | </layout> | ||
| 71 | </widget> | ||
| 72 | </item> | ||
| 73 | <item> | ||
| 74 | <widget class="QGroupBox" name="touch_group_box"> | ||
| 75 | <property name="title"> | ||
| 76 | <string>Touch</string> | ||
| 77 | </property> | ||
| 78 | <layout class="QVBoxLayout"> | ||
| 79 | <item> | ||
| 80 | <layout class="QHBoxLayout"> | ||
| 81 | <item> | ||
| 82 | <widget class="QLabel" name="touch_provider_label"> | ||
| 83 | <property name="text"> | ||
| 84 | <string>Touch Provider:</string> | ||
| 85 | </property> | ||
| 86 | </widget> | ||
| 87 | </item> | ||
| 88 | <item> | ||
| 89 | <widget class="QComboBox" name="touch_provider"/> | ||
| 90 | </item> | ||
| 91 | </layout> | ||
| 92 | </item> | ||
| 93 | <item> | ||
| 94 | <layout class="QHBoxLayout"> | ||
| 95 | <item> | ||
| 96 | <widget class="QLabel" name="touch_calibration_label"> | ||
| 97 | <property name="text"> | ||
| 98 | <string>Calibration:</string> | ||
| 99 | </property> | ||
| 100 | </widget> | ||
| 101 | </item> | ||
| 102 | <item> | ||
| 103 | <widget class="QLabel" name="touch_calibration"> | ||
| 104 | <property name="text"> | ||
| 105 | <string>(100, 50) - (1800, 850)</string> | ||
| 106 | </property> | ||
| 107 | <property name="alignment"> | ||
| 108 | <set>Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter</set> | ||
| 109 | </property> | ||
| 110 | </widget> | ||
| 111 | </item> | ||
| 112 | <item> | ||
| 113 | <widget class="QPushButton" name="touch_calibration_config"> | ||
| 114 | <property name="sizePolicy"> | ||
| 115 | <sizepolicy hsizetype="Maximum" vsizetype="Fixed"> | ||
| 116 | <horstretch>0</horstretch> | ||
| 117 | <verstretch>0</verstretch> | ||
| 118 | </sizepolicy> | ||
| 119 | </property> | ||
| 120 | <property name="text"> | ||
| 121 | <string>Configure</string> | ||
| 122 | </property> | ||
| 123 | </widget> | ||
| 124 | </item> | ||
| 125 | </layout> | ||
| 126 | </item> | ||
| 127 | <item> | ||
| 128 | <layout class="QHBoxLayout"> | ||
| 129 | <item> | ||
| 130 | <widget class="QCheckBox" name="touch_from_button_checkbox"> | ||
| 131 | <property name="sizePolicy"> | ||
| 132 | <sizepolicy hsizetype="Maximum" vsizetype="Fixed"> | ||
| 133 | <horstretch>0</horstretch> | ||
| 134 | <verstretch>0</verstretch> | ||
| 135 | </sizepolicy> | ||
| 136 | </property> | ||
| 137 | <property name="text"> | ||
| 138 | <string>Use button mapping:</string> | ||
| 139 | </property> | ||
| 140 | </widget> | ||
| 141 | </item> | ||
| 142 | <item> | ||
| 143 | <widget class="QComboBox" name="touch_from_button_map"/> | ||
| 144 | </item> | ||
| 145 | <item> | ||
| 146 | <widget class="QPushButton" name="touch_from_button_config_btn"> | ||
| 147 | <property name="sizePolicy"> | ||
| 148 | <sizepolicy hsizetype="Maximum" vsizetype="Fixed"> | ||
| 149 | <horstretch>0</horstretch> | ||
| 150 | <verstretch>0</verstretch> | ||
| 151 | </sizepolicy> | ||
| 152 | </property> | ||
| 153 | <property name="text"> | ||
| 154 | <string>Configure</string> | ||
| 155 | </property> | ||
| 156 | </widget> | ||
| 157 | </item> | ||
| 158 | </layout> | ||
| 159 | </item> | ||
| 160 | </layout> | ||
| 161 | </widget> | ||
| 162 | </item> | ||
| 163 | <item> | ||
| 164 | <widget class="QGroupBox" name="udp_config_group_box"> | ||
| 165 | <property name="title"> | ||
| 166 | <string>CemuhookUDP Config</string> | ||
| 167 | </property> | ||
| 168 | <layout class="QVBoxLayout"> | ||
| 169 | <item> | ||
| 170 | <widget class="QLabel" name="udp_help"> | ||
| 171 | <property name="text"> | ||
| 172 | <string>You may use any Cemuhook compatible UDP input source to provide motion and touch input.</string> | ||
| 173 | </property> | ||
| 174 | <property name="alignment"> | ||
| 175 | <set>Qt::AlignLeading|Qt::AlignLeft|Qt::AlignTop</set> | ||
| 176 | </property> | ||
| 177 | <property name="wordWrap"> | ||
| 178 | <bool>true</bool> | ||
| 179 | </property> | ||
| 180 | </widget> | ||
| 181 | </item> | ||
| 182 | <item> | ||
| 183 | <layout class="QHBoxLayout"> | ||
| 184 | <item> | ||
| 185 | <widget class="QLabel" name="udp_server_label"> | ||
| 186 | <property name="text"> | ||
| 187 | <string>Server:</string> | ||
| 188 | </property> | ||
| 189 | </widget> | ||
| 190 | </item> | ||
| 191 | <item> | ||
| 192 | <widget class="QLineEdit" name="udp_server"> | ||
| 193 | <property name="sizePolicy"> | ||
| 194 | <sizepolicy hsizetype="Minimum" vsizetype="Fixed"> | ||
| 195 | <horstretch>0</horstretch> | ||
| 196 | <verstretch>0</verstretch> | ||
| 197 | </sizepolicy> | ||
| 198 | </property> | ||
| 199 | </widget> | ||
| 200 | </item> | ||
| 201 | </layout> | ||
| 202 | </item> | ||
| 203 | <item> | ||
| 204 | <layout class="QHBoxLayout"> | ||
| 205 | <item> | ||
| 206 | <widget class="QLabel" name="udp_port_label"> | ||
| 207 | <property name="text"> | ||
| 208 | <string>Port:</string> | ||
| 209 | </property> | ||
| 210 | </widget> | ||
| 211 | </item> | ||
| 212 | <item> | ||
| 213 | <widget class="QLineEdit" name="udp_port"> | ||
| 214 | <property name="sizePolicy"> | ||
| 215 | <sizepolicy hsizetype="Minimum" vsizetype="Fixed"> | ||
| 216 | <horstretch>0</horstretch> | ||
| 217 | <verstretch>0</verstretch> | ||
| 218 | </sizepolicy> | ||
| 219 | </property> | ||
| 220 | </widget> | ||
| 221 | </item> | ||
| 222 | </layout> | ||
| 223 | </item> | ||
| 224 | <item> | ||
| 225 | <layout class="QHBoxLayout"> | ||
| 226 | <item> | ||
| 227 | <widget class="QLabel" name="udp_pad_index_label"> | ||
| 228 | <property name="text"> | ||
| 229 | <string>Pad:</string> | ||
| 230 | </property> | ||
| 231 | </widget> | ||
| 232 | </item> | ||
| 233 | <item> | ||
| 234 | <widget class="QComboBox" name="udp_pad_index"> | ||
| 235 | <item> | ||
| 236 | <property name="text"> | ||
| 237 | <string>Pad 1</string> | ||
| 238 | </property> | ||
| 239 | </item> | ||
| 240 | <item> | ||
| 241 | <property name="text"> | ||
| 242 | <string>Pad 2</string> | ||
| 243 | </property> | ||
| 244 | </item> | ||
| 245 | <item> | ||
| 246 | <property name="text"> | ||
| 247 | <string>Pad 3</string> | ||
| 248 | </property> | ||
| 249 | </item> | ||
| 250 | <item> | ||
| 251 | <property name="text"> | ||
| 252 | <string>Pad 4</string> | ||
| 253 | </property> | ||
| 254 | </item> | ||
| 255 | </widget> | ||
| 256 | </item> | ||
| 257 | </layout> | ||
| 258 | </item> | ||
| 259 | <item> | ||
| 260 | <layout class="QHBoxLayout"> | ||
| 261 | <item> | ||
| 262 | <widget class="QLabel" name="udp_learn_more"> | ||
| 263 | <property name="text"> | ||
| 264 | <string>Learn More</string> | ||
| 265 | </property> | ||
| 266 | </widget> | ||
| 267 | </item> | ||
| 268 | <item> | ||
| 269 | <widget class="QPushButton" name="udp_test"> | ||
| 270 | <property name="sizePolicy"> | ||
| 271 | <sizepolicy hsizetype="Maximum" vsizetype="Fixed"> | ||
| 272 | <horstretch>0</horstretch> | ||
| 273 | <verstretch>0</verstretch> | ||
| 274 | </sizepolicy> | ||
| 275 | </property> | ||
| 276 | <property name="text"> | ||
| 277 | <string>Test</string> | ||
| 278 | </property> | ||
| 279 | </widget> | ||
| 280 | </item> | ||
| 281 | </layout> | ||
| 282 | </item> | ||
| 283 | </layout> | ||
| 284 | </widget> | ||
| 285 | </item> | ||
| 286 | <item> | ||
| 287 | <spacer> | ||
| 288 | <property name="orientation"> | ||
| 289 | <enum>Qt::Vertical</enum> | ||
| 290 | </property> | ||
| 291 | <property name="sizeHint" stdset="0"> | ||
| 292 | <size> | ||
| 293 | <width>167</width> | ||
| 294 | <height>55</height> | ||
| 295 | </size> | ||
| 296 | </property> | ||
| 297 | </spacer> | ||
| 298 | </item> | ||
| 299 | <item> | ||
| 300 | <widget class="QDialogButtonBox" name="buttonBox"> | ||
| 301 | <property name="standardButtons"> | ||
| 302 | <set>QDialogButtonBox::Cancel|QDialogButtonBox::Ok</set> | ||
| 303 | </property> | ||
| 304 | </widget> | ||
| 305 | </item> | ||
| 306 | </layout> | ||
| 307 | </widget> | ||
| 308 | <resources/> | ||
| 309 | <connections> | ||
| 310 | <connection> | ||
| 311 | <sender>buttonBox</sender> | ||
| 312 | <signal>accepted()</signal> | ||
| 313 | <receiver>ConfigureMotionTouch</receiver> | ||
| 314 | <slot>ApplyConfiguration()</slot> | ||
| 315 | <hints> | ||
| 316 | <hint type="sourcelabel"> | ||
| 317 | <x>220</x> | ||
| 318 | <y>380</y> | ||
| 319 | </hint> | ||
| 320 | <hint type="destinationlabel"> | ||
| 321 | <x>220</x> | ||
| 322 | <y>200</y> | ||
| 323 | </hint> | ||
| 324 | </hints> | ||
| 325 | </connection> | ||
| 326 | </connections> | ||
| 327 | </ui> | ||
diff --git a/src/yuzu/configuration/configure_touch_from_button.cpp b/src/yuzu/configuration/configure_touch_from_button.cpp new file mode 100644 index 000000000..15557e4b8 --- /dev/null +++ b/src/yuzu/configuration/configure_touch_from_button.cpp | |||
| @@ -0,0 +1,623 @@ | |||
| 1 | // Copyright 2020 Citra Emulator Project | ||
| 2 | // Licensed under GPLv2 or any later version | ||
| 3 | // Refer to the license.txt file included. | ||
| 4 | |||
| 5 | #include <QInputDialog> | ||
| 6 | #include <QKeyEvent> | ||
| 7 | #include <QMessageBox> | ||
| 8 | #include <QMouseEvent> | ||
| 9 | #include <QResizeEvent> | ||
| 10 | #include <QStandardItemModel> | ||
| 11 | #include <QTimer> | ||
| 12 | #include "common/param_package.h" | ||
| 13 | #include "core/frontend/framebuffer_layout.h" | ||
| 14 | #include "core/settings.h" | ||
| 15 | #include "input_common/main.h" | ||
| 16 | #include "ui_configure_touch_from_button.h" | ||
| 17 | #include "yuzu/configuration/configure_touch_from_button.h" | ||
| 18 | #include "yuzu/configuration/configure_touch_widget.h" | ||
| 19 | |||
| 20 | static QString GetKeyName(int key_code) { | ||
| 21 | switch (key_code) { | ||
| 22 | case Qt::Key_Shift: | ||
| 23 | return QObject::tr("Shift"); | ||
| 24 | case Qt::Key_Control: | ||
| 25 | return QObject::tr("Ctrl"); | ||
| 26 | case Qt::Key_Alt: | ||
| 27 | return QObject::tr("Alt"); | ||
| 28 | case Qt::Key_Meta: | ||
| 29 | return QString{}; | ||
| 30 | default: | ||
| 31 | return QKeySequence(key_code).toString(); | ||
| 32 | } | ||
| 33 | } | ||
| 34 | |||
| 35 | static QString ButtonToText(const Common::ParamPackage& param) { | ||
| 36 | if (!param.Has("engine")) { | ||
| 37 | return QObject::tr("[not set]"); | ||
| 38 | } | ||
| 39 | |||
| 40 | if (param.Get("engine", "") == "keyboard") { | ||
| 41 | return GetKeyName(param.Get("code", 0)); | ||
| 42 | } | ||
| 43 | |||
| 44 | if (param.Get("engine", "") == "sdl") { | ||
| 45 | if (param.Has("hat")) { | ||
| 46 | const QString hat_str = QString::fromStdString(param.Get("hat", "")); | ||
| 47 | const QString direction_str = QString::fromStdString(param.Get("direction", "")); | ||
| 48 | |||
| 49 | return QObject::tr("Hat %1 %2").arg(hat_str, direction_str); | ||
| 50 | } | ||
| 51 | |||
| 52 | if (param.Has("axis")) { | ||
| 53 | const QString axis_str = QString::fromStdString(param.Get("axis", "")); | ||
| 54 | const QString direction_str = QString::fromStdString(param.Get("direction", "")); | ||
| 55 | |||
| 56 | return QObject::tr("Axis %1%2").arg(axis_str, direction_str); | ||
| 57 | } | ||
| 58 | |||
| 59 | if (param.Has("button")) { | ||
| 60 | const QString button_str = QString::fromStdString(param.Get("button", "")); | ||
| 61 | |||
| 62 | return QObject::tr("Button %1").arg(button_str); | ||
| 63 | } | ||
| 64 | |||
| 65 | return {}; | ||
| 66 | } | ||
| 67 | |||
| 68 | return QObject::tr("[unknown]"); | ||
| 69 | } | ||
| 70 | |||
| 71 | ConfigureTouchFromButton::ConfigureTouchFromButton( | ||
| 72 | QWidget* parent, const std::vector<Settings::TouchFromButtonMap>& touch_maps, | ||
| 73 | InputCommon::InputSubsystem* input_subsystem_, const int default_index) | ||
| 74 | : QDialog(parent), ui(std::make_unique<Ui::ConfigureTouchFromButton>()), | ||
| 75 | touch_maps(touch_maps), input_subsystem{input_subsystem_}, selected_index(default_index), | ||
| 76 | timeout_timer(std::make_unique<QTimer>()), poll_timer(std::make_unique<QTimer>()) { | ||
| 77 | ui->setupUi(this); | ||
| 78 | binding_list_model = new QStandardItemModel(0, 3, this); | ||
| 79 | binding_list_model->setHorizontalHeaderLabels( | ||
| 80 | {tr("Button"), tr("X", "X axis"), tr("Y", "Y axis")}); | ||
| 81 | ui->binding_list->setModel(binding_list_model); | ||
| 82 | ui->bottom_screen->SetCoordLabel(ui->coord_label); | ||
| 83 | |||
| 84 | SetConfiguration(); | ||
| 85 | UpdateUiDisplay(); | ||
| 86 | ConnectEvents(); | ||
| 87 | } | ||
| 88 | |||
| 89 | ConfigureTouchFromButton::~ConfigureTouchFromButton() = default; | ||
| 90 | |||
| 91 | void ConfigureTouchFromButton::showEvent(QShowEvent* ev) { | ||
| 92 | QWidget::showEvent(ev); | ||
| 93 | |||
| 94 | // width values are not valid in the constructor | ||
| 95 | const int w = | ||
| 96 | ui->binding_list->viewport()->contentsRect().width() / binding_list_model->columnCount(); | ||
| 97 | if (w <= 0) { | ||
| 98 | return; | ||
| 99 | } | ||
| 100 | ui->binding_list->setColumnWidth(0, w); | ||
| 101 | ui->binding_list->setColumnWidth(1, w); | ||
| 102 | ui->binding_list->setColumnWidth(2, w); | ||
| 103 | } | ||
| 104 | |||
| 105 | void ConfigureTouchFromButton::SetConfiguration() { | ||
| 106 | for (const auto& touch_map : touch_maps) { | ||
| 107 | ui->mapping->addItem(QString::fromStdString(touch_map.name)); | ||
| 108 | } | ||
| 109 | |||
| 110 | ui->mapping->setCurrentIndex(selected_index); | ||
| 111 | } | ||
| 112 | |||
| 113 | void ConfigureTouchFromButton::UpdateUiDisplay() { | ||
| 114 | ui->button_delete->setEnabled(touch_maps.size() > 1); | ||
| 115 | ui->button_delete_bind->setEnabled(false); | ||
| 116 | |||
| 117 | binding_list_model->removeRows(0, binding_list_model->rowCount()); | ||
| 118 | |||
| 119 | for (const auto& button_str : touch_maps[selected_index].buttons) { | ||
| 120 | Common::ParamPackage package{button_str}; | ||
| 121 | QStandardItem* button = new QStandardItem(ButtonToText(package)); | ||
| 122 | button->setData(QString::fromStdString(button_str)); | ||
| 123 | button->setEditable(false); | ||
| 124 | QStandardItem* xcoord = new QStandardItem(QString::number(package.Get("x", 0))); | ||
| 125 | QStandardItem* ycoord = new QStandardItem(QString::number(package.Get("y", 0))); | ||
| 126 | binding_list_model->appendRow({button, xcoord, ycoord}); | ||
| 127 | |||
| 128 | const int dot = ui->bottom_screen->AddDot(package.Get("x", 0), package.Get("y", 0)); | ||
| 129 | button->setData(dot, DataRoleDot); | ||
| 130 | } | ||
| 131 | } | ||
| 132 | |||
| 133 | void ConfigureTouchFromButton::ConnectEvents() { | ||
| 134 | connect(ui->mapping, qOverload<int>(&QComboBox::currentIndexChanged), this, [this](int index) { | ||
| 135 | SaveCurrentMapping(); | ||
| 136 | selected_index = index; | ||
| 137 | UpdateUiDisplay(); | ||
| 138 | }); | ||
| 139 | connect(ui->button_new, &QPushButton::clicked, this, &ConfigureTouchFromButton::NewMapping); | ||
| 140 | connect(ui->button_delete, &QPushButton::clicked, this, | ||
| 141 | &ConfigureTouchFromButton::DeleteMapping); | ||
| 142 | connect(ui->button_rename, &QPushButton::clicked, this, | ||
| 143 | &ConfigureTouchFromButton::RenameMapping); | ||
| 144 | connect(ui->button_delete_bind, &QPushButton::clicked, this, | ||
| 145 | &ConfigureTouchFromButton::DeleteBinding); | ||
| 146 | connect(ui->binding_list, &QTreeView::doubleClicked, this, | ||
| 147 | &ConfigureTouchFromButton::EditBinding); | ||
| 148 | connect(ui->binding_list->selectionModel(), &QItemSelectionModel::selectionChanged, this, | ||
| 149 | &ConfigureTouchFromButton::OnBindingSelection); | ||
| 150 | connect(binding_list_model, &QStandardItemModel::itemChanged, this, | ||
| 151 | &ConfigureTouchFromButton::OnBindingChanged); | ||
| 152 | connect(ui->binding_list->model(), &QStandardItemModel::rowsAboutToBeRemoved, this, | ||
| 153 | &ConfigureTouchFromButton::OnBindingDeleted); | ||
| 154 | connect(ui->bottom_screen, &TouchScreenPreview::DotAdded, this, | ||
| 155 | &ConfigureTouchFromButton::NewBinding); | ||
| 156 | connect(ui->bottom_screen, &TouchScreenPreview::DotSelected, this, | ||
| 157 | &ConfigureTouchFromButton::SetActiveBinding); | ||
| 158 | connect(ui->bottom_screen, &TouchScreenPreview::DotMoved, this, | ||
| 159 | &ConfigureTouchFromButton::SetCoordinates); | ||
| 160 | connect(ui->buttonBox, &QDialogButtonBox::accepted, this, | ||
| 161 | &ConfigureTouchFromButton::ApplyConfiguration); | ||
| 162 | |||
| 163 | connect(timeout_timer.get(), &QTimer::timeout, [this]() { SetPollingResult({}, true); }); | ||
| 164 | |||
| 165 | connect(poll_timer.get(), &QTimer::timeout, [this]() { | ||
| 166 | Common::ParamPackage params; | ||
| 167 | for (auto& poller : device_pollers) { | ||
| 168 | params = poller->GetNextInput(); | ||
| 169 | if (params.Has("engine")) { | ||
| 170 | SetPollingResult(params, false); | ||
| 171 | return; | ||
| 172 | } | ||
| 173 | } | ||
| 174 | }); | ||
| 175 | } | ||
| 176 | |||
| 177 | void ConfigureTouchFromButton::SaveCurrentMapping() { | ||
| 178 | auto& map = touch_maps[selected_index]; | ||
| 179 | map.buttons.clear(); | ||
| 180 | for (int i = 0, rc = binding_list_model->rowCount(); i < rc; ++i) { | ||
| 181 | const auto bind_str = binding_list_model->index(i, 0) | ||
| 182 | .data(Qt::ItemDataRole::UserRole + 1) | ||
| 183 | .toString() | ||
| 184 | .toStdString(); | ||
| 185 | if (bind_str.empty()) { | ||
| 186 | continue; | ||
| 187 | } | ||
| 188 | Common::ParamPackage params{bind_str}; | ||
| 189 | if (!params.Has("engine")) { | ||
| 190 | continue; | ||
| 191 | } | ||
| 192 | params.Set("x", binding_list_model->index(i, 1).data().toInt()); | ||
| 193 | params.Set("y", binding_list_model->index(i, 2).data().toInt()); | ||
| 194 | map.buttons.emplace_back(params.Serialize()); | ||
| 195 | } | ||
| 196 | } | ||
| 197 | |||
| 198 | void ConfigureTouchFromButton::NewMapping() { | ||
| 199 | const QString name = | ||
| 200 | QInputDialog::getText(this, tr("New Profile"), tr("Enter the name for the new profile.")); | ||
| 201 | if (name.isEmpty()) { | ||
| 202 | return; | ||
| 203 | } | ||
| 204 | touch_maps.emplace_back(Settings::TouchFromButtonMap{name.toStdString(), {}}); | ||
| 205 | ui->mapping->addItem(name); | ||
| 206 | ui->mapping->setCurrentIndex(ui->mapping->count() - 1); | ||
| 207 | } | ||
| 208 | |||
| 209 | void ConfigureTouchFromButton::DeleteMapping() { | ||
| 210 | const auto answer = QMessageBox::question( | ||
| 211 | this, tr("Delete Profile"), tr("Delete profile %1?").arg(ui->mapping->currentText())); | ||
| 212 | if (answer != QMessageBox::Yes) { | ||
| 213 | return; | ||
| 214 | } | ||
| 215 | const bool blocked = ui->mapping->blockSignals(true); | ||
| 216 | ui->mapping->removeItem(selected_index); | ||
| 217 | ui->mapping->blockSignals(blocked); | ||
| 218 | touch_maps.erase(touch_maps.begin() + selected_index); | ||
| 219 | selected_index = ui->mapping->currentIndex(); | ||
| 220 | UpdateUiDisplay(); | ||
| 221 | } | ||
| 222 | |||
| 223 | void ConfigureTouchFromButton::RenameMapping() { | ||
| 224 | const QString new_name = QInputDialog::getText(this, tr("Rename Profile"), tr("New name:")); | ||
| 225 | if (new_name.isEmpty()) { | ||
| 226 | return; | ||
| 227 | } | ||
| 228 | ui->mapping->setItemText(selected_index, new_name); | ||
| 229 | touch_maps[selected_index].name = new_name.toStdString(); | ||
| 230 | } | ||
| 231 | |||
| 232 | void ConfigureTouchFromButton::GetButtonInput(const int row_index, const bool is_new) { | ||
| 233 | binding_list_model->item(row_index, 0)->setText(tr("[press key]")); | ||
| 234 | |||
| 235 | input_setter = [this, row_index, is_new](const Common::ParamPackage& params, | ||
| 236 | const bool cancel) { | ||
| 237 | auto* cell = binding_list_model->item(row_index, 0); | ||
| 238 | if (cancel) { | ||
| 239 | if (is_new) { | ||
| 240 | binding_list_model->removeRow(row_index); | ||
| 241 | } else { | ||
| 242 | cell->setText( | ||
| 243 | ButtonToText(Common::ParamPackage{cell->data().toString().toStdString()})); | ||
| 244 | } | ||
| 245 | } else { | ||
| 246 | cell->setText(ButtonToText(params)); | ||
| 247 | cell->setData(QString::fromStdString(params.Serialize())); | ||
| 248 | } | ||
| 249 | }; | ||
| 250 | |||
| 251 | device_pollers = input_subsystem->GetPollers(InputCommon::Polling::DeviceType::Button); | ||
| 252 | |||
| 253 | for (auto& poller : device_pollers) { | ||
| 254 | poller->Start(); | ||
| 255 | } | ||
| 256 | |||
| 257 | grabKeyboard(); | ||
| 258 | grabMouse(); | ||
| 259 | qApp->setOverrideCursor(QCursor(Qt::CursorShape::ArrowCursor)); | ||
| 260 | timeout_timer->start(5000); // Cancel after 5 seconds | ||
| 261 | poll_timer->start(200); // Check for new inputs every 200ms | ||
| 262 | } | ||
| 263 | |||
| 264 | void ConfigureTouchFromButton::NewBinding(const QPoint& pos) { | ||
| 265 | auto* button = new QStandardItem(); | ||
| 266 | button->setEditable(false); | ||
| 267 | auto* x_coord = new QStandardItem(QString::number(pos.x())); | ||
| 268 | auto* y_coord = new QStandardItem(QString::number(pos.y())); | ||
| 269 | |||
| 270 | const int dot_id = ui->bottom_screen->AddDot(pos.x(), pos.y()); | ||
| 271 | button->setData(dot_id, DataRoleDot); | ||
| 272 | |||
| 273 | binding_list_model->appendRow({button, x_coord, y_coord}); | ||
| 274 | ui->binding_list->setFocus(); | ||
| 275 | ui->binding_list->setCurrentIndex(button->index()); | ||
| 276 | |||
| 277 | GetButtonInput(binding_list_model->rowCount() - 1, true); | ||
| 278 | } | ||
| 279 | |||
| 280 | void ConfigureTouchFromButton::EditBinding(const QModelIndex& qi) { | ||
| 281 | if (qi.row() >= 0 && qi.column() == 0) { | ||
| 282 | GetButtonInput(qi.row(), false); | ||
| 283 | } | ||
| 284 | } | ||
| 285 | |||
| 286 | void ConfigureTouchFromButton::DeleteBinding() { | ||
| 287 | const int row_index = ui->binding_list->currentIndex().row(); | ||
| 288 | if (row_index < 0) { | ||
| 289 | return; | ||
| 290 | } | ||
| 291 | ui->bottom_screen->RemoveDot(binding_list_model->index(row_index, 0).data(DataRoleDot).toInt()); | ||
| 292 | binding_list_model->removeRow(row_index); | ||
| 293 | } | ||
| 294 | |||
| 295 | void ConfigureTouchFromButton::OnBindingSelection(const QItemSelection& selected, | ||
| 296 | const QItemSelection& deselected) { | ||
| 297 | ui->button_delete_bind->setEnabled(!selected.isEmpty()); | ||
| 298 | if (!selected.isEmpty()) { | ||
| 299 | const auto dot_data = selected.indexes().first().data(DataRoleDot); | ||
| 300 | if (dot_data.isValid()) { | ||
| 301 | ui->bottom_screen->HighlightDot(dot_data.toInt()); | ||
| 302 | } | ||
| 303 | } | ||
| 304 | if (!deselected.isEmpty()) { | ||
| 305 | const auto dot_data = deselected.indexes().first().data(DataRoleDot); | ||
| 306 | if (dot_data.isValid()) { | ||
| 307 | ui->bottom_screen->HighlightDot(dot_data.toInt(), false); | ||
| 308 | } | ||
| 309 | } | ||
| 310 | } | ||
| 311 | |||
| 312 | void ConfigureTouchFromButton::OnBindingChanged(QStandardItem* item) { | ||
| 313 | if (item->column() == 0) { | ||
| 314 | return; | ||
| 315 | } | ||
| 316 | |||
| 317 | const bool blocked = binding_list_model->blockSignals(true); | ||
| 318 | item->setText(QString::number( | ||
| 319 | std::clamp(item->text().toInt(), 0, | ||
| 320 | static_cast<int>((item->column() == 1 ? Layout::ScreenUndocked::Width | ||
| 321 | : Layout::ScreenUndocked::Height) - | ||
| 322 | 1)))); | ||
| 323 | binding_list_model->blockSignals(blocked); | ||
| 324 | |||
| 325 | const auto dot_data = binding_list_model->index(item->row(), 0).data(DataRoleDot); | ||
| 326 | if (dot_data.isValid()) { | ||
| 327 | ui->bottom_screen->MoveDot(dot_data.toInt(), | ||
| 328 | binding_list_model->item(item->row(), 1)->text().toInt(), | ||
| 329 | binding_list_model->item(item->row(), 2)->text().toInt()); | ||
| 330 | } | ||
| 331 | } | ||
| 332 | |||
| 333 | void ConfigureTouchFromButton::OnBindingDeleted(const QModelIndex& parent, int first, int last) { | ||
| 334 | for (int i = first; i <= last; ++i) { | ||
| 335 | const auto ix = binding_list_model->index(i, 0); | ||
| 336 | if (!ix.isValid()) { | ||
| 337 | return; | ||
| 338 | } | ||
| 339 | const auto dot_data = ix.data(DataRoleDot); | ||
| 340 | if (dot_data.isValid()) { | ||
| 341 | ui->bottom_screen->RemoveDot(dot_data.toInt()); | ||
| 342 | } | ||
| 343 | } | ||
| 344 | } | ||
| 345 | |||
| 346 | void ConfigureTouchFromButton::SetActiveBinding(const int dot_id) { | ||
| 347 | for (int i = 0; i < binding_list_model->rowCount(); ++i) { | ||
| 348 | if (binding_list_model->index(i, 0).data(DataRoleDot) == dot_id) { | ||
| 349 | ui->binding_list->setCurrentIndex(binding_list_model->index(i, 0)); | ||
| 350 | ui->binding_list->setFocus(); | ||
| 351 | return; | ||
| 352 | } | ||
| 353 | } | ||
| 354 | } | ||
| 355 | |||
| 356 | void ConfigureTouchFromButton::SetCoordinates(const int dot_id, const QPoint& pos) { | ||
| 357 | for (int i = 0; i < binding_list_model->rowCount(); ++i) { | ||
| 358 | if (binding_list_model->item(i, 0)->data(DataRoleDot) == dot_id) { | ||
| 359 | binding_list_model->item(i, 1)->setText(QString::number(pos.x())); | ||
| 360 | binding_list_model->item(i, 2)->setText(QString::number(pos.y())); | ||
| 361 | return; | ||
| 362 | } | ||
| 363 | } | ||
| 364 | } | ||
| 365 | |||
| 366 | void ConfigureTouchFromButton::SetPollingResult(const Common::ParamPackage& params, | ||
| 367 | const bool cancel) { | ||
| 368 | releaseKeyboard(); | ||
| 369 | releaseMouse(); | ||
| 370 | qApp->restoreOverrideCursor(); | ||
| 371 | timeout_timer->stop(); | ||
| 372 | poll_timer->stop(); | ||
| 373 | for (auto& poller : device_pollers) { | ||
| 374 | poller->Stop(); | ||
| 375 | } | ||
| 376 | if (input_setter) { | ||
| 377 | (*input_setter)(params, cancel); | ||
| 378 | input_setter.reset(); | ||
| 379 | } | ||
| 380 | } | ||
| 381 | |||
| 382 | void ConfigureTouchFromButton::keyPressEvent(QKeyEvent* event) { | ||
| 383 | if (!input_setter && event->key() == Qt::Key_Delete) { | ||
| 384 | DeleteBinding(); | ||
| 385 | return; | ||
| 386 | } | ||
| 387 | |||
| 388 | if (!input_setter) { | ||
| 389 | return QDialog::keyPressEvent(event); | ||
| 390 | } | ||
| 391 | |||
| 392 | if (event->key() != Qt::Key_Escape) { | ||
| 393 | SetPollingResult(Common::ParamPackage{InputCommon::GenerateKeyboardParam(event->key())}, | ||
| 394 | false); | ||
| 395 | } else { | ||
| 396 | SetPollingResult({}, true); | ||
| 397 | } | ||
| 398 | } | ||
| 399 | |||
| 400 | void ConfigureTouchFromButton::ApplyConfiguration() { | ||
| 401 | SaveCurrentMapping(); | ||
| 402 | accept(); | ||
| 403 | } | ||
| 404 | |||
| 405 | int ConfigureTouchFromButton::GetSelectedIndex() const { | ||
| 406 | return selected_index; | ||
| 407 | } | ||
| 408 | |||
| 409 | std::vector<Settings::TouchFromButtonMap> ConfigureTouchFromButton::GetMaps() const { | ||
| 410 | return touch_maps; | ||
| 411 | } | ||
| 412 | |||
| 413 | TouchScreenPreview::TouchScreenPreview(QWidget* parent) : QFrame(parent) { | ||
| 414 | setBackgroundRole(QPalette::ColorRole::Base); | ||
| 415 | } | ||
| 416 | |||
| 417 | TouchScreenPreview::~TouchScreenPreview() = default; | ||
| 418 | |||
| 419 | void TouchScreenPreview::SetCoordLabel(QLabel* const label) { | ||
| 420 | coord_label = label; | ||
| 421 | } | ||
| 422 | |||
| 423 | int TouchScreenPreview::AddDot(const int device_x, const int device_y) { | ||
| 424 | QFont dot_font{QStringLiteral("monospace")}; | ||
| 425 | dot_font.setStyleHint(QFont::Monospace); | ||
| 426 | dot_font.setPointSize(20); | ||
| 427 | |||
| 428 | auto* dot = new QLabel(this); | ||
| 429 | dot->setAttribute(Qt::WA_TranslucentBackground); | ||
| 430 | dot->setFont(dot_font); | ||
| 431 | dot->setText(QChar(0xD7)); // U+00D7 Multiplication Sign | ||
| 432 | dot->setAlignment(Qt::AlignmentFlag::AlignCenter); | ||
| 433 | dot->setProperty(PropId, ++max_dot_id); | ||
| 434 | dot->setProperty(PropX, device_x); | ||
| 435 | dot->setProperty(PropY, device_y); | ||
| 436 | dot->setCursor(Qt::CursorShape::PointingHandCursor); | ||
| 437 | dot->setMouseTracking(true); | ||
| 438 | dot->installEventFilter(this); | ||
| 439 | dot->show(); | ||
| 440 | PositionDot(dot, device_x, device_y); | ||
| 441 | dots.emplace_back(max_dot_id, dot); | ||
| 442 | return max_dot_id; | ||
| 443 | } | ||
| 444 | |||
| 445 | void TouchScreenPreview::RemoveDot(const int id) { | ||
| 446 | const auto iter = std::find_if(dots.begin(), dots.end(), | ||
| 447 | [id](const auto& entry) { return entry.first == id; }); | ||
| 448 | if (iter == dots.cend()) { | ||
| 449 | return; | ||
| 450 | } | ||
| 451 | |||
| 452 | iter->second->deleteLater(); | ||
| 453 | dots.erase(iter); | ||
| 454 | } | ||
| 455 | |||
| 456 | void TouchScreenPreview::HighlightDot(const int id, const bool active) const { | ||
| 457 | for (const auto& dot : dots) { | ||
| 458 | if (dot.first == id) { | ||
| 459 | // use color property from the stylesheet, or fall back to the default palette | ||
| 460 | if (dot_highlight_color.isValid()) { | ||
| 461 | dot.second->setStyleSheet( | ||
| 462 | active ? QStringLiteral("color: %1").arg(dot_highlight_color.name()) | ||
| 463 | : QString{}); | ||
| 464 | } else { | ||
| 465 | dot.second->setForegroundRole(active ? QPalette::ColorRole::LinkVisited | ||
| 466 | : QPalette::ColorRole::NoRole); | ||
| 467 | } | ||
| 468 | if (active) { | ||
| 469 | dot.second->raise(); | ||
| 470 | } | ||
| 471 | return; | ||
| 472 | } | ||
| 473 | } | ||
| 474 | } | ||
| 475 | |||
| 476 | void TouchScreenPreview::MoveDot(const int id, const int device_x, const int device_y) const { | ||
| 477 | const auto iter = std::find_if(dots.begin(), dots.end(), | ||
| 478 | [id](const auto& entry) { return entry.first == id; }); | ||
| 479 | if (iter == dots.cend()) { | ||
| 480 | return; | ||
| 481 | } | ||
| 482 | |||
| 483 | iter->second->setProperty(PropX, device_x); | ||
| 484 | iter->second->setProperty(PropY, device_y); | ||
| 485 | PositionDot(iter->second, device_x, device_y); | ||
| 486 | } | ||
| 487 | |||
| 488 | void TouchScreenPreview::resizeEvent(QResizeEvent* event) { | ||
| 489 | if (ignore_resize) { | ||
| 490 | return; | ||
| 491 | } | ||
| 492 | |||
| 493 | const int target_width = std::min(width(), height() * 4 / 3); | ||
| 494 | const int target_height = std::min(height(), width() * 3 / 4); | ||
| 495 | if (target_width == width() && target_height == height()) { | ||
| 496 | return; | ||
| 497 | } | ||
| 498 | ignore_resize = true; | ||
| 499 | setGeometry((parentWidget()->contentsRect().width() - target_width) / 2, y(), target_width, | ||
| 500 | target_height); | ||
| 501 | ignore_resize = false; | ||
| 502 | |||
| 503 | if (event->oldSize().width() != target_width || event->oldSize().height() != target_height) { | ||
| 504 | for (const auto& dot : dots) { | ||
| 505 | PositionDot(dot.second); | ||
| 506 | } | ||
| 507 | } | ||
| 508 | } | ||
| 509 | |||
| 510 | void TouchScreenPreview::mouseMoveEvent(QMouseEvent* event) { | ||
| 511 | if (!coord_label) { | ||
| 512 | return; | ||
| 513 | } | ||
| 514 | const auto pos = MapToDeviceCoords(event->x(), event->y()); | ||
| 515 | if (pos) { | ||
| 516 | coord_label->setText(QStringLiteral("X: %1, Y: %2").arg(pos->x()).arg(pos->y())); | ||
| 517 | } else { | ||
| 518 | coord_label->clear(); | ||
| 519 | } | ||
| 520 | } | ||
| 521 | |||
| 522 | void TouchScreenPreview::leaveEvent(QEvent* event) { | ||
| 523 | if (coord_label) { | ||
| 524 | coord_label->clear(); | ||
| 525 | } | ||
| 526 | } | ||
| 527 | |||
| 528 | void TouchScreenPreview::mousePressEvent(QMouseEvent* event) { | ||
| 529 | if (event->button() != Qt::MouseButton::LeftButton) { | ||
| 530 | return; | ||
| 531 | } | ||
| 532 | const auto pos = MapToDeviceCoords(event->x(), event->y()); | ||
| 533 | if (pos) { | ||
| 534 | emit DotAdded(*pos); | ||
| 535 | } | ||
| 536 | } | ||
| 537 | |||
| 538 | bool TouchScreenPreview::eventFilter(QObject* obj, QEvent* event) { | ||
| 539 | switch (event->type()) { | ||
| 540 | case QEvent::Type::MouseButtonPress: { | ||
| 541 | const auto mouse_event = static_cast<QMouseEvent*>(event); | ||
| 542 | if (mouse_event->button() != Qt::MouseButton::LeftButton) { | ||
| 543 | break; | ||
| 544 | } | ||
| 545 | emit DotSelected(obj->property(PropId).toInt()); | ||
| 546 | |||
| 547 | drag_state.dot = qobject_cast<QLabel*>(obj); | ||
| 548 | drag_state.start_pos = mouse_event->globalPos(); | ||
| 549 | return true; | ||
| 550 | } | ||
| 551 | case QEvent::Type::MouseMove: { | ||
| 552 | if (!drag_state.dot) { | ||
| 553 | break; | ||
| 554 | } | ||
| 555 | const auto mouse_event = static_cast<QMouseEvent*>(event); | ||
| 556 | if (!drag_state.active) { | ||
| 557 | drag_state.active = | ||
| 558 | (mouse_event->globalPos() - drag_state.start_pos).manhattanLength() >= | ||
| 559 | QApplication::startDragDistance(); | ||
| 560 | if (!drag_state.active) { | ||
| 561 | break; | ||
| 562 | } | ||
| 563 | } | ||
| 564 | auto current_pos = mapFromGlobal(mouse_event->globalPos()); | ||
| 565 | current_pos.setX(std::clamp(current_pos.x(), contentsMargins().left(), | ||
| 566 | contentsMargins().left() + contentsRect().width() - 1)); | ||
| 567 | current_pos.setY(std::clamp(current_pos.y(), contentsMargins().top(), | ||
| 568 | contentsMargins().top() + contentsRect().height() - 1)); | ||
| 569 | const auto device_coord = MapToDeviceCoords(current_pos.x(), current_pos.y()); | ||
| 570 | if (device_coord) { | ||
| 571 | drag_state.dot->setProperty(PropX, device_coord->x()); | ||
| 572 | drag_state.dot->setProperty(PropY, device_coord->y()); | ||
| 573 | PositionDot(drag_state.dot, device_coord->x(), device_coord->y()); | ||
| 574 | emit DotMoved(drag_state.dot->property(PropId).toInt(), *device_coord); | ||
| 575 | if (coord_label) { | ||
| 576 | coord_label->setText( | ||
| 577 | QStringLiteral("X: %1, Y: %2").arg(device_coord->x()).arg(device_coord->y())); | ||
| 578 | } | ||
| 579 | } | ||
| 580 | return true; | ||
| 581 | } | ||
| 582 | case QEvent::Type::MouseButtonRelease: { | ||
| 583 | drag_state.dot.clear(); | ||
| 584 | drag_state.active = false; | ||
| 585 | return true; | ||
| 586 | } | ||
| 587 | default: | ||
| 588 | break; | ||
| 589 | } | ||
| 590 | return obj->eventFilter(obj, event); | ||
| 591 | } | ||
| 592 | |||
| 593 | std::optional<QPoint> TouchScreenPreview::MapToDeviceCoords(const int screen_x, | ||
| 594 | const int screen_y) const { | ||
| 595 | const float t_x = 0.5f + static_cast<float>(screen_x - contentsMargins().left()) * | ||
| 596 | (Layout::ScreenUndocked::Width - 1) / (contentsRect().width() - 1); | ||
| 597 | const float t_y = 0.5f + static_cast<float>(screen_y - contentsMargins().top()) * | ||
| 598 | (Layout::ScreenUndocked::Height - 1) / | ||
| 599 | (contentsRect().height() - 1); | ||
| 600 | if (t_x >= 0.5f && t_x < Layout::ScreenUndocked::Width && t_y >= 0.5f && | ||
| 601 | t_y < Layout::ScreenUndocked::Height) { | ||
| 602 | |||
| 603 | return QPoint{static_cast<int>(t_x), static_cast<int>(t_y)}; | ||
| 604 | } | ||
| 605 | return std::nullopt; | ||
| 606 | } | ||
| 607 | |||
| 608 | void TouchScreenPreview::PositionDot(QLabel* const dot, const int device_x, | ||
| 609 | const int device_y) const { | ||
| 610 | const float device_coord_x = | ||
| 611 | static_cast<float>(device_x >= 0 ? device_x : dot->property(PropX).toInt()); | ||
| 612 | int x_coord = static_cast<int>( | ||
| 613 | device_coord_x * (contentsRect().width() - 1) / (Layout::ScreenUndocked::Width - 1) + | ||
| 614 | contentsMargins().left() - static_cast<float>(dot->width()) / 2 + 0.5f); | ||
| 615 | |||
| 616 | const float device_coord_y = | ||
| 617 | static_cast<float>(device_y >= 0 ? device_y : dot->property(PropY).toInt()); | ||
| 618 | const int y_coord = static_cast<int>( | ||
| 619 | device_coord_y * (contentsRect().height() - 1) / (Layout::ScreenUndocked::Height - 1) + | ||
| 620 | contentsMargins().top() - static_cast<float>(dot->height()) / 2 + 0.5f); | ||
| 621 | |||
| 622 | dot->move(x_coord, y_coord); | ||
| 623 | } | ||
diff --git a/src/yuzu/configuration/configure_touch_from_button.h b/src/yuzu/configuration/configure_touch_from_button.h new file mode 100644 index 000000000..d9513e3bc --- /dev/null +++ b/src/yuzu/configuration/configure_touch_from_button.h | |||
| @@ -0,0 +1,92 @@ | |||
| 1 | // Copyright 2020 Citra Emulator Project | ||
| 2 | // Licensed under GPLv2 or any later version | ||
| 3 | // Refer to the license.txt file included. | ||
| 4 | |||
| 5 | #pragma once | ||
| 6 | |||
| 7 | #include <functional> | ||
| 8 | #include <memory> | ||
| 9 | #include <optional> | ||
| 10 | #include <vector> | ||
| 11 | #include <QDialog> | ||
| 12 | |||
| 13 | class QItemSelection; | ||
| 14 | class QModelIndex; | ||
| 15 | class QStandardItemModel; | ||
| 16 | class QStandardItem; | ||
| 17 | class QTimer; | ||
| 18 | |||
| 19 | namespace Common { | ||
| 20 | class ParamPackage; | ||
| 21 | } | ||
| 22 | |||
| 23 | namespace InputCommon { | ||
| 24 | class InputSubsystem; | ||
| 25 | } | ||
| 26 | |||
| 27 | namespace InputCommon::Polling { | ||
| 28 | class DevicePoller; | ||
| 29 | } | ||
| 30 | |||
| 31 | namespace Settings { | ||
| 32 | struct TouchFromButtonMap; | ||
| 33 | } | ||
| 34 | |||
| 35 | namespace Ui { | ||
| 36 | class ConfigureTouchFromButton; | ||
| 37 | } | ||
| 38 | |||
| 39 | class ConfigureTouchFromButton : public QDialog { | ||
| 40 | Q_OBJECT | ||
| 41 | |||
| 42 | public: | ||
| 43 | explicit ConfigureTouchFromButton(QWidget* parent, | ||
| 44 | const std::vector<Settings::TouchFromButtonMap>& touch_maps, | ||
| 45 | InputCommon::InputSubsystem* input_subsystem_, | ||
| 46 | int default_index = 0); | ||
| 47 | ~ConfigureTouchFromButton() override; | ||
| 48 | |||
| 49 | int GetSelectedIndex() const; | ||
| 50 | std::vector<Settings::TouchFromButtonMap> GetMaps() const; | ||
| 51 | |||
| 52 | public slots: | ||
| 53 | void ApplyConfiguration(); | ||
| 54 | void NewBinding(const QPoint& pos); | ||
| 55 | void SetActiveBinding(int dot_id); | ||
| 56 | void SetCoordinates(int dot_id, const QPoint& pos); | ||
| 57 | |||
| 58 | protected: | ||
| 59 | void showEvent(QShowEvent* ev) override; | ||
| 60 | void keyPressEvent(QKeyEvent* event) override; | ||
| 61 | |||
| 62 | private slots: | ||
| 63 | void NewMapping(); | ||
| 64 | void DeleteMapping(); | ||
| 65 | void RenameMapping(); | ||
| 66 | void EditBinding(const QModelIndex& qi); | ||
| 67 | void DeleteBinding(); | ||
| 68 | void OnBindingSelection(const QItemSelection& selected, const QItemSelection& deselected); | ||
| 69 | void OnBindingChanged(QStandardItem* item); | ||
| 70 | void OnBindingDeleted(const QModelIndex& parent, int first, int last); | ||
| 71 | |||
| 72 | private: | ||
| 73 | void SetConfiguration(); | ||
| 74 | void UpdateUiDisplay(); | ||
| 75 | void ConnectEvents(); | ||
| 76 | void GetButtonInput(int row_index, bool is_new); | ||
| 77 | void SetPollingResult(const Common::ParamPackage& params, bool cancel); | ||
| 78 | void SaveCurrentMapping(); | ||
| 79 | |||
| 80 | std::unique_ptr<Ui::ConfigureTouchFromButton> ui; | ||
| 81 | std::vector<Settings::TouchFromButtonMap> touch_maps; | ||
| 82 | QStandardItemModel* binding_list_model; | ||
| 83 | InputCommon::InputSubsystem* input_subsystem; | ||
| 84 | int selected_index; | ||
| 85 | |||
| 86 | std::unique_ptr<QTimer> timeout_timer; | ||
| 87 | std::unique_ptr<QTimer> poll_timer; | ||
| 88 | std::vector<std::unique_ptr<InputCommon::Polling::DevicePoller>> device_pollers; | ||
| 89 | std::optional<std::function<void(const Common::ParamPackage&, bool)>> input_setter; | ||
| 90 | |||
| 91 | static constexpr int DataRoleDot = Qt::ItemDataRole::UserRole + 2; | ||
| 92 | }; | ||
diff --git a/src/yuzu/configuration/configure_touch_from_button.ui b/src/yuzu/configuration/configure_touch_from_button.ui new file mode 100644 index 000000000..f581e27e0 --- /dev/null +++ b/src/yuzu/configuration/configure_touch_from_button.ui | |||
| @@ -0,0 +1,231 @@ | |||
| 1 | <?xml version="1.0" encoding="UTF-8"?> | ||
| 2 | <ui version="4.0"> | ||
| 3 | <class>ConfigureTouchFromButton</class> | ||
| 4 | <widget class="QDialog" name="ConfigureTouchFromButton"> | ||
| 5 | <property name="geometry"> | ||
| 6 | <rect> | ||
| 7 | <x>0</x> | ||
| 8 | <y>0</y> | ||
| 9 | <width>500</width> | ||
| 10 | <height>500</height> | ||
| 11 | </rect> | ||
| 12 | </property> | ||
| 13 | <property name="windowTitle"> | ||
| 14 | <string>Configure Touchscreen Mappings</string> | ||
| 15 | </property> | ||
| 16 | <layout class="QVBoxLayout"> | ||
| 17 | <item> | ||
| 18 | <layout class="QHBoxLayout" name="horizontalLayout"> | ||
| 19 | <item> | ||
| 20 | <widget class="QLabel" name="label"> | ||
| 21 | <property name="text"> | ||
| 22 | <string>Mapping:</string> | ||
| 23 | </property> | ||
| 24 | <property name="textFormat"> | ||
| 25 | <enum>Qt::PlainText</enum> | ||
| 26 | </property> | ||
| 27 | </widget> | ||
| 28 | </item> | ||
| 29 | <item> | ||
| 30 | <widget class="QComboBox" name="mapping"> | ||
| 31 | <property name="sizePolicy"> | ||
| 32 | <sizepolicy hsizetype="MinimumExpanding" vsizetype="Fixed"> | ||
| 33 | <horstretch>0</horstretch> | ||
| 34 | <verstretch>0</verstretch> | ||
| 35 | </sizepolicy> | ||
| 36 | </property> | ||
| 37 | </widget> | ||
| 38 | </item> | ||
| 39 | <item> | ||
| 40 | <widget class="QPushButton" name="button_new"> | ||
| 41 | <property name="sizePolicy"> | ||
| 42 | <sizepolicy hsizetype="Maximum" vsizetype="Fixed"> | ||
| 43 | <horstretch>0</horstretch> | ||
| 44 | <verstretch>0</verstretch> | ||
| 45 | </sizepolicy> | ||
| 46 | </property> | ||
| 47 | <property name="text"> | ||
| 48 | <string>New</string> | ||
| 49 | </property> | ||
| 50 | </widget> | ||
| 51 | </item> | ||
| 52 | <item> | ||
| 53 | <widget class="QPushButton" name="button_delete"> | ||
| 54 | <property name="sizePolicy"> | ||
| 55 | <sizepolicy hsizetype="Maximum" vsizetype="Fixed"> | ||
| 56 | <horstretch>0</horstretch> | ||
| 57 | <verstretch>0</verstretch> | ||
| 58 | </sizepolicy> | ||
| 59 | </property> | ||
| 60 | <property name="text"> | ||
| 61 | <string>Delete</string> | ||
| 62 | </property> | ||
| 63 | </widget> | ||
| 64 | </item> | ||
| 65 | <item> | ||
| 66 | <widget class="QPushButton" name="button_rename"> | ||
| 67 | <property name="sizePolicy"> | ||
| 68 | <sizepolicy hsizetype="Maximum" vsizetype="Fixed"> | ||
| 69 | <horstretch>0</horstretch> | ||
| 70 | <verstretch>0</verstretch> | ||
| 71 | </sizepolicy> | ||
| 72 | </property> | ||
| 73 | <property name="text"> | ||
| 74 | <string>Rename</string> | ||
| 75 | </property> | ||
| 76 | </widget> | ||
| 77 | </item> | ||
| 78 | </layout> | ||
| 79 | </item> | ||
| 80 | <item> | ||
| 81 | <widget class="Line" name="line"> | ||
| 82 | <property name="orientation"> | ||
| 83 | <enum>Qt::Horizontal</enum> | ||
| 84 | </property> | ||
| 85 | </widget> | ||
| 86 | </item> | ||
| 87 | <item> | ||
| 88 | <layout class="QHBoxLayout" name="horizontalLayout_2"> | ||
| 89 | <item> | ||
| 90 | <widget class="QLabel" name="label_2"> | ||
| 91 | <property name="text"> | ||
| 92 | <string>Click the bottom area to add a point, then press a button to bind. | ||
| 93 | Drag points to change position, or double-click table cells to edit values.</string> | ||
| 94 | </property> | ||
| 95 | <property name="textFormat"> | ||
| 96 | <enum>Qt::PlainText</enum> | ||
| 97 | </property> | ||
| 98 | </widget> | ||
| 99 | </item> | ||
| 100 | <item> | ||
| 101 | <spacer name="horizontalSpacer"> | ||
| 102 | <property name="orientation"> | ||
| 103 | <enum>Qt::Horizontal</enum> | ||
| 104 | </property> | ||
| 105 | <property name="sizeHint" stdset="0"> | ||
| 106 | <size> | ||
| 107 | <width>40</width> | ||
| 108 | <height>20</height> | ||
| 109 | </size> | ||
| 110 | </property> | ||
| 111 | </spacer> | ||
| 112 | </item> | ||
| 113 | <item> | ||
| 114 | <widget class="QPushButton" name="button_delete_bind"> | ||
| 115 | <property name="text"> | ||
| 116 | <string>Delete Point</string> | ||
| 117 | </property> | ||
| 118 | </widget> | ||
| 119 | </item> | ||
| 120 | </layout> | ||
| 121 | </item> | ||
| 122 | <item> | ||
| 123 | <widget class="QTreeView" name="binding_list"> | ||
| 124 | <property name="sizePolicy"> | ||
| 125 | <sizepolicy hsizetype="Expanding" vsizetype="Expanding"> | ||
| 126 | <horstretch>0</horstretch> | ||
| 127 | <verstretch>0</verstretch> | ||
| 128 | </sizepolicy> | ||
| 129 | </property> | ||
| 130 | <property name="rootIsDecorated"> | ||
| 131 | <bool>false</bool> | ||
| 132 | </property> | ||
| 133 | <property name="uniformRowHeights"> | ||
| 134 | <bool>true</bool> | ||
| 135 | </property> | ||
| 136 | <property name="itemsExpandable"> | ||
| 137 | <bool>false</bool> | ||
| 138 | </property> | ||
| 139 | </widget> | ||
| 140 | </item> | ||
| 141 | <item> | ||
| 142 | <widget class="TouchScreenPreview" name="bottom_screen"> | ||
| 143 | <property name="sizePolicy"> | ||
| 144 | <sizepolicy hsizetype="Expanding" vsizetype="Expanding"> | ||
| 145 | <horstretch>0</horstretch> | ||
| 146 | <verstretch>0</verstretch> | ||
| 147 | </sizepolicy> | ||
| 148 | </property> | ||
| 149 | <property name="minimumSize"> | ||
| 150 | <size> | ||
| 151 | <width>160</width> | ||
| 152 | <height>120</height> | ||
| 153 | </size> | ||
| 154 | </property> | ||
| 155 | <property name="baseSize"> | ||
| 156 | <size> | ||
| 157 | <width>320</width> | ||
| 158 | <height>240</height> | ||
| 159 | </size> | ||
| 160 | </property> | ||
| 161 | <property name="cursor"> | ||
| 162 | <cursorShape>CrossCursor</cursorShape> | ||
| 163 | </property> | ||
| 164 | <property name="mouseTracking"> | ||
| 165 | <bool>true</bool> | ||
| 166 | </property> | ||
| 167 | <property name="autoFillBackground"> | ||
| 168 | <bool>true</bool> | ||
| 169 | </property> | ||
| 170 | <property name="frameShape"> | ||
| 171 | <enum>QFrame::StyledPanel</enum> | ||
| 172 | </property> | ||
| 173 | <property name="frameShadow"> | ||
| 174 | <enum>QFrame::Sunken</enum> | ||
| 175 | </property> | ||
| 176 | </widget> | ||
| 177 | </item> | ||
| 178 | <item> | ||
| 179 | <layout class="QHBoxLayout" name="horizontalLayout_3"> | ||
| 180 | <item> | ||
| 181 | <widget class="QLabel" name="coord_label"> | ||
| 182 | <property name="sizePolicy"> | ||
| 183 | <sizepolicy hsizetype="Expanding" vsizetype="Preferred"> | ||
| 184 | <horstretch>0</horstretch> | ||
| 185 | <verstretch>0</verstretch> | ||
| 186 | </sizepolicy> | ||
| 187 | </property> | ||
| 188 | <property name="textFormat"> | ||
| 189 | <enum>Qt::PlainText</enum> | ||
| 190 | </property> | ||
| 191 | </widget> | ||
| 192 | </item> | ||
| 193 | <item> | ||
| 194 | <widget class="QDialogButtonBox" name="buttonBox"> | ||
| 195 | <property name="standardButtons"> | ||
| 196 | <set>QDialogButtonBox::Cancel|QDialogButtonBox::Ok</set> | ||
| 197 | </property> | ||
| 198 | </widget> | ||
| 199 | </item> | ||
| 200 | </layout> | ||
| 201 | </item> | ||
| 202 | </layout> | ||
| 203 | </widget> | ||
| 204 | <customwidgets> | ||
| 205 | <customwidget> | ||
| 206 | <class>TouchScreenPreview</class> | ||
| 207 | <extends>QFrame</extends> | ||
| 208 | <header>yuzu/configuration/configure_touch_widget.h</header> | ||
| 209 | <container>1</container> | ||
| 210 | </customwidget> | ||
| 211 | </customwidgets> | ||
| 212 | <resources/> | ||
| 213 | <connections> | ||
| 214 | <connection> | ||
| 215 | <sender>buttonBox</sender> | ||
| 216 | <signal>rejected()</signal> | ||
| 217 | <receiver>ConfigureTouchFromButton</receiver> | ||
| 218 | <slot>reject()</slot> | ||
| 219 | <hints> | ||
| 220 | <hint type="sourcelabel"> | ||
| 221 | <x>249</x> | ||
| 222 | <y>428</y> | ||
| 223 | </hint> | ||
| 224 | <hint type="destinationlabel"> | ||
| 225 | <x>249</x> | ||
| 226 | <y>224</y> | ||
| 227 | </hint> | ||
| 228 | </hints> | ||
| 229 | </connection> | ||
| 230 | </connections> | ||
| 231 | </ui> | ||
diff --git a/src/yuzu/configuration/configure_touch_widget.h b/src/yuzu/configuration/configure_touch_widget.h new file mode 100644 index 000000000..347b46583 --- /dev/null +++ b/src/yuzu/configuration/configure_touch_widget.h | |||
| @@ -0,0 +1,62 @@ | |||
| 1 | // Copyright 2020 Citra Emulator Project | ||
| 2 | // Licensed under GPLv2 or any later version | ||
| 3 | // Refer to the license.txt file included. | ||
| 4 | |||
| 5 | #pragma once | ||
| 6 | |||
| 7 | #include <optional> | ||
| 8 | #include <utility> | ||
| 9 | #include <vector> | ||
| 10 | #include <QFrame> | ||
| 11 | #include <QPointer> | ||
| 12 | |||
| 13 | class QLabel; | ||
| 14 | |||
| 15 | // Widget for representing touchscreen coordinates | ||
| 16 | class TouchScreenPreview : public QFrame { | ||
| 17 | Q_OBJECT | ||
| 18 | Q_PROPERTY(QColor dotHighlightColor MEMBER dot_highlight_color) | ||
| 19 | |||
| 20 | public: | ||
| 21 | explicit TouchScreenPreview(QWidget* parent); | ||
| 22 | ~TouchScreenPreview() override; | ||
| 23 | |||
| 24 | void SetCoordLabel(QLabel*); | ||
| 25 | int AddDot(int device_x, int device_y); | ||
| 26 | void RemoveDot(int id); | ||
| 27 | void HighlightDot(int id, bool active = true) const; | ||
| 28 | void MoveDot(int id, int device_x, int device_y) const; | ||
| 29 | |||
| 30 | signals: | ||
| 31 | void DotAdded(const QPoint& pos); | ||
| 32 | void DotSelected(int dot_id); | ||
| 33 | void DotMoved(int dot_id, const QPoint& pos); | ||
| 34 | |||
| 35 | protected: | ||
| 36 | void resizeEvent(QResizeEvent*) override; | ||
| 37 | void mouseMoveEvent(QMouseEvent*) override; | ||
| 38 | void leaveEvent(QEvent*) override; | ||
| 39 | void mousePressEvent(QMouseEvent*) override; | ||
| 40 | bool eventFilter(QObject*, QEvent*) override; | ||
| 41 | |||
| 42 | private: | ||
| 43 | std::optional<QPoint> MapToDeviceCoords(int screen_x, int screen_y) const; | ||
| 44 | void PositionDot(QLabel* dot, int device_x = -1, int device_y = -1) const; | ||
| 45 | |||
| 46 | bool ignore_resize = false; | ||
| 47 | QPointer<QLabel> coord_label; | ||
| 48 | |||
| 49 | std::vector<std::pair<int, QLabel*>> dots; | ||
| 50 | int max_dot_id = 0; | ||
| 51 | QColor dot_highlight_color; | ||
| 52 | static constexpr char PropId[] = "dot_id"; | ||
| 53 | static constexpr char PropX[] = "device_x"; | ||
| 54 | static constexpr char PropY[] = "device_y"; | ||
| 55 | |||
| 56 | struct DragState { | ||
| 57 | bool active = false; | ||
| 58 | QPointer<QLabel> dot; | ||
| 59 | QPoint start_pos; | ||
| 60 | }; | ||
| 61 | DragState drag_state; | ||
| 62 | }; | ||
diff --git a/src/yuzu/main.cpp b/src/yuzu/main.cpp index cab9d680a..a1b61d119 100644 --- a/src/yuzu/main.cpp +++ b/src/yuzu/main.cpp | |||
| @@ -187,7 +187,7 @@ static void InitializeLogging() { | |||
| 187 | } | 187 | } |
| 188 | 188 | ||
| 189 | GMainWindow::GMainWindow() | 189 | GMainWindow::GMainWindow() |
| 190 | : input_subsystem{std::make_unique<InputCommon::InputSubsystem>()}, | 190 | : input_subsystem{std::make_shared<InputCommon::InputSubsystem>()}, |
| 191 | config{std::make_unique<Config>()}, vfs{std::make_shared<FileSys::RealVfsFilesystem>()}, | 191 | config{std::make_unique<Config>()}, vfs{std::make_shared<FileSys::RealVfsFilesystem>()}, |
| 192 | provider{std::make_unique<FileSys::ManualContentProvider>()} { | 192 | provider{std::make_unique<FileSys::ManualContentProvider>()} { |
| 193 | InitializeLogging(); | 193 | InitializeLogging(); |
| @@ -474,7 +474,7 @@ void GMainWindow::InitializeWidgets() { | |||
| 474 | #ifdef YUZU_ENABLE_COMPATIBILITY_REPORTING | 474 | #ifdef YUZU_ENABLE_COMPATIBILITY_REPORTING |
| 475 | ui.action_Report_Compatibility->setVisible(true); | 475 | ui.action_Report_Compatibility->setVisible(true); |
| 476 | #endif | 476 | #endif |
| 477 | render_window = new GRenderWindow(this, emu_thread.get(), input_subsystem.get()); | 477 | render_window = new GRenderWindow(this, emu_thread.get(), input_subsystem); |
| 478 | render_window->hide(); | 478 | render_window->hide(); |
| 479 | 479 | ||
| 480 | game_list = new GameList(vfs, provider.get(), this); | 480 | game_list = new GameList(vfs, provider.get(), this); |
diff --git a/src/yuzu/main.h b/src/yuzu/main.h index 957f20fa8..0ce66a1ca 100644 --- a/src/yuzu/main.h +++ b/src/yuzu/main.h | |||
| @@ -258,7 +258,7 @@ private: | |||
| 258 | Ui::MainWindow ui; | 258 | Ui::MainWindow ui; |
| 259 | 259 | ||
| 260 | std::unique_ptr<DiscordRPC::DiscordInterface> discord_rpc; | 260 | std::unique_ptr<DiscordRPC::DiscordInterface> discord_rpc; |
| 261 | std::unique_ptr<InputCommon::InputSubsystem> input_subsystem; | 261 | std::shared_ptr<InputCommon::InputSubsystem> input_subsystem; |
| 262 | 262 | ||
| 263 | GRenderWindow* render_window; | 263 | GRenderWindow* render_window; |
| 264 | GameList* game_list; | 264 | GameList* game_list; |
diff --git a/src/yuzu/main.ui b/src/yuzu/main.ui index 87ea985d8..2f3792247 100644 --- a/src/yuzu/main.ui +++ b/src/yuzu/main.ui | |||
| @@ -293,7 +293,7 @@ | |||
| 293 | <bool>false</bool> | 293 | <bool>false</bool> |
| 294 | </property> | 294 | </property> |
| 295 | <property name="text"> | 295 | <property name="text"> |
| 296 | <string>Configure Current Game..</string> | 296 | <string>Configure Current Game...</string> |
| 297 | </property> | 297 | </property> |
| 298 | </action> | 298 | </action> |
| 299 | </widget> | 299 | </widget> |