early-access version 1344
This commit is contained in:
		| @@ -1,7 +1,7 @@ | ||||
| yuzu emulator early access | ||||
| ============= | ||||
|  | ||||
| This is the source code for early-access 1343. | ||||
| This is the source code for early-access 1344. | ||||
|  | ||||
| ## Legal Notice | ||||
|  | ||||
|   | ||||
| @@ -399,8 +399,6 @@ add_library(core STATIC | ||||
|     hle/service/ldr/ldr.h | ||||
|     hle/service/lm/lm.cpp | ||||
|     hle/service/lm/lm.h | ||||
|     hle/service/lm/manager.cpp | ||||
|     hle/service/lm/manager.h | ||||
|     hle/service/mig/mig.cpp | ||||
|     hle/service/mig/mig.h | ||||
|     hle/service/mii/manager.cpp | ||||
|   | ||||
| @@ -36,7 +36,6 @@ | ||||
| #include "core/hle/service/apm/controller.h" | ||||
| #include "core/hle/service/filesystem/filesystem.h" | ||||
| #include "core/hle/service/glue/manager.h" | ||||
| #include "core/hle/service/lm/manager.h" | ||||
| #include "core/hle/service/service.h" | ||||
| #include "core/hle/service/sm/sm.h" | ||||
| #include "core/hle/service/time/time_manager.h" | ||||
| @@ -293,8 +292,6 @@ struct System::Impl { | ||||
|                                         perf_stats->GetMeanFrametime()); | ||||
|         } | ||||
|  | ||||
|         lm_manager.Flush(); | ||||
|  | ||||
|         is_powered_on = false; | ||||
|         exit_lock = false; | ||||
|  | ||||
| @@ -398,7 +395,6 @@ struct System::Impl { | ||||
|  | ||||
|     /// Service State | ||||
|     Service::Glue::ARPManager arp_manager; | ||||
|     Service::LM::Manager lm_manager{reporter}; | ||||
|     Service::Time::TimeManager time_manager; | ||||
|  | ||||
|     /// Service manager | ||||
| @@ -720,14 +716,6 @@ const Service::APM::Controller& System::GetAPMController() const { | ||||
|     return impl->apm_controller; | ||||
| } | ||||
|  | ||||
| Service::LM::Manager& System::GetLogManager() { | ||||
|     return impl->lm_manager; | ||||
| } | ||||
|  | ||||
| const Service::LM::Manager& System::GetLogManager() const { | ||||
|     return impl->lm_manager; | ||||
| } | ||||
|  | ||||
| Service::Time::TimeManager& System::GetTimeManager() { | ||||
|     return impl->time_manager; | ||||
| } | ||||
|   | ||||
| @@ -62,10 +62,6 @@ namespace Glue { | ||||
| class ARPManager; | ||||
| } | ||||
|  | ||||
| namespace LM { | ||||
| class Manager; | ||||
| } // namespace LM | ||||
|  | ||||
| namespace SM { | ||||
| class ServiceManager; | ||||
| } // namespace SM | ||||
| @@ -351,9 +347,6 @@ public: | ||||
|     [[nodiscard]] Service::APM::Controller& GetAPMController(); | ||||
|     [[nodiscard]] const Service::APM::Controller& GetAPMController() const; | ||||
|  | ||||
|     [[nodiscard]] Service::LM::Manager& GetLogManager(); | ||||
|     [[nodiscard]] const Service::LM::Manager& GetLogManager() const; | ||||
|  | ||||
|     [[nodiscard]] Service::Time::TimeManager& GetTimeManager(); | ||||
|     [[nodiscard]] const Service::Time::TimeManager& GetTimeManager() const; | ||||
|  | ||||
|   | ||||
| @@ -5,22 +5,71 @@ | ||||
| #include <sstream> | ||||
| #include <string> | ||||
|  | ||||
| #include <optional> | ||||
| #include <unordered_map> | ||||
| #include <boost/container_hash/hash.hpp> | ||||
| #include "common/logging/log.h" | ||||
| #include "common/scope_exit.h" | ||||
| #include "core/core.h" | ||||
| #include "core/hle/ipc_helpers.h" | ||||
| #include "core/hle/service/lm/lm.h" | ||||
| #include "core/hle/service/lm/manager.h" | ||||
| #include "core/hle/service/service.h" | ||||
| #include "core/memory.h" | ||||
|  | ||||
| namespace Service::LM { | ||||
| enum class LogSeverity : u8 { | ||||
|     Trace = 0, | ||||
|     Info = 1, | ||||
|     Warning = 2, | ||||
|     Error = 3, | ||||
|     Fatal = 4, | ||||
| }; | ||||
|  | ||||
| // To keep flags out of hashing as well as the payload size | ||||
| struct LogPacketHeaderEntrty { | ||||
|     u64_le pid{}; | ||||
|     u64_le tid{}; | ||||
|     LogSeverity severity{}; | ||||
|     u8 verbosity{}; | ||||
|  | ||||
|     auto operator<=>(const LogPacketHeaderEntrty&) const = default; | ||||
| }; | ||||
| } // namespace Service::LM | ||||
|  | ||||
| namespace std { | ||||
| template <> | ||||
| struct hash<Service::LM::LogPacketHeaderEntrty> { | ||||
|     std::size_t operator()(const Service::LM::LogPacketHeaderEntrty& k) const { | ||||
|         std::size_t seed{}; | ||||
|         boost::hash_combine(seed, k.pid); | ||||
|         boost::hash_combine(seed, k.tid); | ||||
|         boost::hash_combine(seed, k.severity); | ||||
|         boost::hash_combine(seed, k.verbosity); | ||||
|         return seed; | ||||
|     }; | ||||
| }; | ||||
| } // namespace std | ||||
|  | ||||
| namespace Service::LM { | ||||
|  | ||||
| enum class LogDestination : u32 { | ||||
|     TargetManager = 1 << 0, | ||||
|     Uart = 1 << 1, | ||||
|     UartSleep = 1 << 2, | ||||
|     All = 0xffff, | ||||
| }; | ||||
| DECLARE_ENUM_FLAG_OPERATORS(LogDestination); | ||||
|  | ||||
| enum class LogPacketFlags : u8 { | ||||
|     Head = 1 << 0, | ||||
|     Tail = 1 << 1, | ||||
|     LittleEndian = 1 << 2, | ||||
| }; | ||||
| DECLARE_ENUM_FLAG_OPERATORS(LogPacketFlags); | ||||
|  | ||||
| class ILogger final : public ServiceFramework<ILogger> { | ||||
| public: | ||||
|     explicit ILogger(Core::System& system_) | ||||
|         : ServiceFramework{system_, "ILogger"}, manager{system_.GetLogManager()}, | ||||
|           memory{system_.Memory()} { | ||||
|     explicit ILogger(Core::System& system_) : ServiceFramework{system_, "ILogger"} { | ||||
|         static const FunctionInfo functions[] = { | ||||
|             {0, &ILogger::Log, "Log"}, | ||||
|             {1, &ILogger::SetDestination, "SetDestination"}, | ||||
| @@ -30,54 +79,257 @@ public: | ||||
|  | ||||
| private: | ||||
|     void Log(Kernel::HLERequestContext& ctx) { | ||||
|         std::size_t offset{}; | ||||
|         const auto data = ctx.ReadBuffer(); | ||||
|  | ||||
|         // This function only succeeds - Get that out of the way | ||||
|         IPC::ResponseBuilder rb{ctx, 2}; | ||||
|         rb.Push(RESULT_SUCCESS); | ||||
|  | ||||
|         // Read MessageHeader, despite not doing anything with it right now | ||||
|         MessageHeader header{}; | ||||
|         VAddr addr{ctx.BufferDescriptorX()[0].Address()}; | ||||
|         const VAddr end_addr{addr + ctx.BufferDescriptorX()[0].size}; | ||||
|         memory.ReadBlock(addr, &header, sizeof(MessageHeader)); | ||||
|         addr += sizeof(MessageHeader); | ||||
|  | ||||
|         FieldMap fields; | ||||
|         while (addr < end_addr) { | ||||
|             const auto field = static_cast<Field>(memory.Read8(addr++)); | ||||
|             const auto length = memory.Read8(addr++); | ||||
|  | ||||
|             if (static_cast<Field>(memory.Read8(addr)) == Field::Skip) { | ||||
|                 ++addr; | ||||
|             } | ||||
|  | ||||
|             SCOPE_EXIT({ addr += length; }); | ||||
|  | ||||
|             if (field == Field::Skip) { | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             std::vector<u8> data(length); | ||||
|             memory.ReadBlock(addr, data.data(), length); | ||||
|             fields.emplace(field, std::move(data)); | ||||
|         if (data.size() < sizeof(LogPacketHeader)) { | ||||
|             LOG_ERROR(Service_LM, "Data size is too small for header! size={}", data.size()); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         manager.Log({header, std::move(fields)}); | ||||
|         LogPacketHeader header{}; | ||||
|         std::memcpy(&header, data.data(), sizeof(LogPacketHeader)); | ||||
|         offset += sizeof(LogPacketHeader); | ||||
|  | ||||
|         LogPacketHeaderEntrty entry{ | ||||
|             .pid = header.pid, | ||||
|             .tid = header.tid, | ||||
|             .severity = header.severity, | ||||
|             .verbosity = header.verbosity, | ||||
|         }; | ||||
|  | ||||
|         if (True(header.flags & LogPacketFlags::Head)) { | ||||
|             std::vector<u8> tmp(data.size() - sizeof(LogPacketHeader)); | ||||
|             std::memcpy(tmp.data(), data.data() + offset, tmp.size()); | ||||
|             entries[entry] = std::move(tmp); | ||||
|         } else { | ||||
|             // Append to existing entry | ||||
|             if (!entries.contains(entry)) { | ||||
|                 LOG_ERROR(Service_LM, "Log entry does not exist!"); | ||||
|                 return; | ||||
|             } | ||||
|             std::vector<u8> tmp(data.size() - sizeof(LogPacketHeader)); | ||||
|  | ||||
|             auto& existing_entry = entries[entry]; | ||||
|             const auto base = existing_entry.size(); | ||||
|             existing_entry.resize(base + (data.size() - sizeof(LogPacketHeader))); | ||||
|             std::memcpy(existing_entry.data() + base, data.data() + offset, | ||||
|                         (data.size() - sizeof(LogPacketHeader))); | ||||
|         } | ||||
|  | ||||
|         if (True(header.flags & LogPacketFlags::Tail)) { | ||||
|             auto it = entries.find(entry); | ||||
|             if (it == entries.end()) { | ||||
|                 LOG_ERROR(Service_LM, "Log entry does not exist!"); | ||||
|                 return; | ||||
|             } | ||||
|             ParseLog(it->first, it->second); | ||||
|             entries.erase(it); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     void SetDestination(Kernel::HLERequestContext& ctx) { | ||||
|         IPC::RequestParser rp{ctx}; | ||||
|         const auto destination = rp.PopEnum<DestinationFlag>(); | ||||
|         const auto log_destination = rp.PopEnum<LogDestination>(); | ||||
|  | ||||
|         LOG_DEBUG(Service_LM, "called, destination={:08X}", destination); | ||||
|  | ||||
|         manager.SetDestination(destination); | ||||
|         LOG_DEBUG(Service_LM, "called, destination={}", DestinationToString(log_destination)); | ||||
|         destination = log_destination; | ||||
|  | ||||
|         IPC::ResponseBuilder rb{ctx, 2}; | ||||
|         rb.Push(RESULT_SUCCESS); | ||||
|     } | ||||
|  | ||||
|     Manager& manager; | ||||
|     Core::Memory::Memory& memory; | ||||
|     u32 ReadLeb128(const std::vector<u8>& data, std::size_t& offset) { | ||||
|         u32 result{}; | ||||
|         u32 shift{}; | ||||
|         do { | ||||
|             result |= (data[offset] & 0x7f) << shift; | ||||
|             shift += 7; | ||||
|             offset++; | ||||
|             if (offset >= data.size()) { | ||||
|                 break; | ||||
|             } | ||||
|         } while ((data[offset] & 0x80) != 0); | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     std::string ReadString(const std::vector<u8>& data, std::size_t& offset, std::size_t length) { | ||||
|         std::string output(length, '\0'); | ||||
|         output.resize(length); | ||||
|         std::memcpy(output.data(), data.data() + offset, length); | ||||
|         offset += length; | ||||
|         return output; | ||||
|     } | ||||
|  | ||||
|     u32_le ReadAsU32(const std::vector<u8>& data, std::size_t& offset, std::size_t length) { | ||||
|         ASSERT(length == sizeof(u32)); | ||||
|         u32_le output{}; | ||||
|         std::memcpy(&output, data.data() + offset, sizeof(u32)); | ||||
|         offset += length; | ||||
|         return output; | ||||
|     } | ||||
|  | ||||
|     u64_le ReadAsU64(const std::vector<u8>& data, std::size_t& offset, std::size_t length) { | ||||
|         ASSERT(length == sizeof(u64)); | ||||
|         u64_le output{}; | ||||
|         std::memcpy(&output, data.data() + offset, sizeof(u64)); | ||||
|         offset += length; | ||||
|         return output; | ||||
|     } | ||||
|  | ||||
|     void ParseLog(const LogPacketHeaderEntrty entry, const std::vector<u8>& log_data) { | ||||
|         // Possible entries | ||||
|         std::optional<std::string> text_log; | ||||
|         std::optional<u32> line_number; | ||||
|         std::optional<std::string> file_name; | ||||
|         std::optional<std::string> function_name; | ||||
|         std::optional<std::string> module_name; | ||||
|         std::optional<std::string> thread_name; | ||||
|         std::optional<u64> log_pack_drop_count; | ||||
|         std::optional<s64> user_system_clock; | ||||
|         std::optional<std::string> process_name; | ||||
|  | ||||
|         std::size_t offset{}; | ||||
|         while (offset < log_data.size()) { | ||||
|             const auto key = static_cast<LogDataChunkKey>(ReadLeb128(log_data, offset)); | ||||
|             const auto chunk_size = ReadLeb128(log_data, offset); | ||||
|  | ||||
|             switch (key) { | ||||
|             case LogDataChunkKey::LogSessionBegin: | ||||
|             case LogDataChunkKey::LogSessionEnd: | ||||
|                 break; | ||||
|             case LogDataChunkKey::TextLog: | ||||
|                 text_log = ReadString(log_data, offset, chunk_size); | ||||
|                 break; | ||||
|             case LogDataChunkKey::LineNumber: | ||||
|                 line_number = ReadAsU32(log_data, offset, chunk_size); | ||||
|                 break; | ||||
|             case LogDataChunkKey::FileName: | ||||
|                 file_name = ReadString(log_data, offset, chunk_size); | ||||
|                 break; | ||||
|             case LogDataChunkKey::FunctionName: | ||||
|                 function_name = ReadString(log_data, offset, chunk_size); | ||||
|                 break; | ||||
|             case LogDataChunkKey::ModuleName: | ||||
|                 module_name = ReadString(log_data, offset, chunk_size); | ||||
|                 break; | ||||
|             case LogDataChunkKey::ThreadName: | ||||
|                 thread_name = ReadString(log_data, offset, chunk_size); | ||||
|                 break; | ||||
|             case LogDataChunkKey::LogPacketDropCount: | ||||
|                 log_pack_drop_count = ReadAsU64(log_data, offset, chunk_size); | ||||
|                 break; | ||||
|             case LogDataChunkKey::UserSystemClock: | ||||
|                 user_system_clock = ReadAsU64(log_data, offset, chunk_size); | ||||
|                 break; | ||||
|             case LogDataChunkKey::ProcessName: | ||||
|                 process_name = ReadString(log_data, offset, chunk_size); | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         std::string output_log{}; | ||||
|         if (process_name && process_name->empty()) { | ||||
|             output_log += fmt::format("Process: {}\n", *process_name); | ||||
|         } | ||||
|         if (module_name && !module_name->empty()) { | ||||
|             output_log += fmt::format("Module: {}\n", *module_name); | ||||
|         } | ||||
|         if (file_name && !file_name->empty()) { | ||||
|             output_log += fmt::format("File: {}\n", *file_name); | ||||
|         } | ||||
|         if (function_name && !function_name->empty()) { | ||||
|             output_log += fmt::format("Function: {}\n", *function_name); | ||||
|         } | ||||
|         if (line_number && *line_number != 0) { | ||||
|             output_log += fmt::format("Line: {}\n", *line_number); | ||||
|         } | ||||
|         output_log += fmt::format("ProcessID: {}\n", entry.pid); | ||||
|         output_log += fmt::format("ThreadID: {}\n", entry.tid); | ||||
|  | ||||
|         if (text_log && !text_log->empty()) { | ||||
|             output_log += fmt::format("Log Text: {}\n", *text_log); | ||||
|         } | ||||
|  | ||||
|         switch (entry.severity) { | ||||
|         case LogSeverity::Trace: | ||||
|             LOG_DEBUG(Service_LM, "LogManager DEBUG ({}):\n{}", DestinationToString(destination), | ||||
|                       output_log); | ||||
|             break; | ||||
|         case LogSeverity::Info: | ||||
|             LOG_INFO(Service_LM, "LogManager INFO ({}):\n{}", DestinationToString(destination), | ||||
|                      output_log); | ||||
|             break; | ||||
|         case LogSeverity::Warning: | ||||
|             LOG_WARNING(Service_LM, "LogManager WARNING ({}):\n{}", | ||||
|                         DestinationToString(destination), output_log); | ||||
|             break; | ||||
|         case LogSeverity::Error: | ||||
|             LOG_ERROR(Service_LM, "LogManager ERROR ({}):\n{}", DestinationToString(destination), | ||||
|                       output_log); | ||||
|             break; | ||||
|         case LogSeverity::Fatal: | ||||
|             LOG_CRITICAL(Service_LM, "LogManager FATAL ({}):\n{}", DestinationToString(destination), | ||||
|                          output_log); | ||||
|             break; | ||||
|         default: | ||||
|             LOG_CRITICAL(Service_LM, "LogManager UNKNOWN ({}):\n{}", | ||||
|                          DestinationToString(destination), output_log); | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     std::string DestinationToString(LogDestination destination) { | ||||
|         if (True(destination & LogDestination::All)) { | ||||
|             return "TargetManager | Uart | UartSleep"; | ||||
|         } | ||||
|         std::string output{}; | ||||
|         if (True(destination & LogDestination::TargetManager)) { | ||||
|             output += "| TargetManager"; | ||||
|         } | ||||
|         if (True(destination & LogDestination::Uart)) { | ||||
|             output += "| Uart"; | ||||
|         } | ||||
|         if (True(destination & LogDestination::UartSleep)) { | ||||
|             output += "| UartSleep"; | ||||
|         } | ||||
|         if (output.length() > 0) { | ||||
|             return output.substr(2); | ||||
|         } | ||||
|         return "No Destination"; | ||||
|     } | ||||
|  | ||||
|     enum class LogDataChunkKey : u32 { | ||||
|         LogSessionBegin = 0, | ||||
|         LogSessionEnd = 1, | ||||
|         TextLog = 2, | ||||
|         LineNumber = 3, | ||||
|         FileName = 4, | ||||
|         FunctionName = 5, | ||||
|         ModuleName = 6, | ||||
|         ThreadName = 7, | ||||
|         LogPacketDropCount = 8, | ||||
|         UserSystemClock = 9, | ||||
|         ProcessName = 10, | ||||
|     }; | ||||
|  | ||||
|     struct LogPacketHeader { | ||||
|         u64_le pid{}; | ||||
|         u64_le tid{}; | ||||
|         LogPacketFlags flags{}; | ||||
|         INSERT_PADDING_BYTES(1); | ||||
|         LogSeverity severity{}; | ||||
|         u8 verbosity{}; | ||||
|         u32_le payload_size{}; | ||||
|     }; | ||||
|     static_assert(sizeof(LogPacketHeader) == 0x18, "LogPacketHeader is an invalid size"); | ||||
|  | ||||
|     std::unordered_map<LogPacketHeaderEntrty, std::vector<u8>> entries{}; | ||||
|     LogDestination destination{LogDestination::All}; | ||||
| }; | ||||
|  | ||||
| class LM final : public ServiceFramework<LM> { | ||||
|   | ||||
| @@ -20,7 +20,6 @@ | ||||
| #include "core/hle/kernel/memory/page_table.h" | ||||
| #include "core/hle/kernel/process.h" | ||||
| #include "core/hle/result.h" | ||||
| #include "core/hle/service/lm/manager.h" | ||||
| #include "core/memory.h" | ||||
| #include "core/reporter.h" | ||||
| #include "core/settings.h" | ||||
| @@ -360,55 +359,6 @@ void Reporter::SaveErrorReport(u64 title_id, ResultCode result, | ||||
|     SaveToFile(std::move(out), GetPath("error_report", title_id, timestamp)); | ||||
| } | ||||
|  | ||||
| void Reporter::SaveLogReport(u32 destination, std::vector<Service::LM::LogMessage> messages) const { | ||||
|     if (!IsReportingEnabled()) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     const auto timestamp = GetTimestamp(); | ||||
|     json out; | ||||
|  | ||||
|     out["yuzu_version"] = GetYuzuVersionData(); | ||||
|     out["report_common"] = | ||||
|         GetReportCommonData(system.CurrentProcess()->GetTitleID(), RESULT_SUCCESS, timestamp); | ||||
|  | ||||
|     out["log_destination"] = | ||||
|         fmt::format("{}", static_cast<Service::LM::DestinationFlag>(destination)); | ||||
|  | ||||
|     auto json_messages = json::array(); | ||||
|     std::transform(messages.begin(), messages.end(), std::back_inserter(json_messages), | ||||
|                    [](const Service::LM::LogMessage& message) { | ||||
|                        json out; | ||||
|                        out["is_head"] = fmt::format("{}", message.header.IsHeadLog()); | ||||
|                        out["is_tail"] = fmt::format("{}", message.header.IsTailLog()); | ||||
|                        out["pid"] = fmt::format("{:016X}", message.header.pid); | ||||
|                        out["thread_context"] = | ||||
|                            fmt::format("{:016X}", message.header.thread_context); | ||||
|                        out["payload_size"] = fmt::format("{:016X}", message.header.payload_size); | ||||
|                        out["flags"] = fmt::format("{:04X}", message.header.flags.Value()); | ||||
|                        out["severity"] = fmt::format("{}", message.header.severity.Value()); | ||||
|                        out["verbosity"] = fmt::format("{:02X}", message.header.verbosity); | ||||
|  | ||||
|                        auto fields = json::array(); | ||||
|                        std::transform(message.fields.begin(), message.fields.end(), | ||||
|                                       std::back_inserter(fields), [](const auto& kv) { | ||||
|                                           json out; | ||||
|                                           out["type"] = fmt::format("{}", kv.first); | ||||
|                                           out["data"] = | ||||
|                                               Service::LM::FormatField(kv.first, kv.second); | ||||
|                                           return out; | ||||
|                                       }); | ||||
|  | ||||
|                        out["fields"] = std::move(fields); | ||||
|                        return out; | ||||
|                    }); | ||||
|  | ||||
|     out["log_messages"] = std::move(json_messages); | ||||
|  | ||||
|     SaveToFile(std::move(out), | ||||
|                GetPath("log_report", system.CurrentProcess()->GetTitleID(), timestamp)); | ||||
| } | ||||
|  | ||||
| void Reporter::SaveFilesystemAccessReport(Service::FileSystem::LogMode log_mode, | ||||
|                                           std::string log_message) const { | ||||
|     if (!IsReportingEnabled()) | ||||
|   | ||||
| @@ -72,9 +72,6 @@ public: | ||||
|     void SaveFilesystemAccessReport(Service::FileSystem::LogMode log_mode, | ||||
|                                     std::string log_message) const; | ||||
|  | ||||
|     // Used by lm services | ||||
|     void SaveLogReport(u32 destination, std::vector<Service::LM::LogMessage> messages) const; | ||||
|  | ||||
|     // Can be used anywhere to generate a backtrace and general info report at any point during | ||||
|     // execution. Not intended to be used for anything other than debugging or testing. | ||||
|     void SaveUserReport() const; | ||||
|   | ||||
| @@ -1000,7 +1000,8 @@ void Config::SavePlayerValue(std::size_t player_index) { | ||||
|                  static_cast<u8>(Settings::ControllerType::ProController)); | ||||
|  | ||||
|     if (!player_prefix.isEmpty()) { | ||||
|         WriteSetting(QStringLiteral("%1connected").arg(player_prefix), player.connected, false); | ||||
|         WriteSetting(QStringLiteral("%1connected").arg(player_prefix), player.connected, | ||||
|                      player_index == 0); | ||||
|         WriteSetting(QStringLiteral("%1vibration_enabled").arg(player_prefix), | ||||
|                      player.vibration_enabled, true); | ||||
|         WriteSetting(QStringLiteral("%1vibration_strength").arg(player_prefix), | ||||
|   | ||||
| @@ -585,6 +585,16 @@ void ConfigureInputPlayer::ApplyConfiguration() { | ||||
|  | ||||
|     std::transform(motions_param.begin(), motions_param.end(), motions.begin(), | ||||
|                    [](const Common::ParamPackage& param) { return param.Serialize(); }); | ||||
|  | ||||
|     // Apply configuration for handheld | ||||
|     if (player_index == 0) { | ||||
|         auto& handheld = Settings::values.players.GetValue()[HANDHELD_INDEX]; | ||||
|         if (player.controller_type == Settings::ControllerType::Handheld) { | ||||
|             handheld = player; | ||||
|         } | ||||
|         handheld.connected = ui->groupConnectedController->isChecked() && | ||||
|                              player.controller_type == Settings::ControllerType::Handheld; | ||||
|     } | ||||
| } | ||||
|  | ||||
| void ConfigureInputPlayer::TryConnectSelectedController() { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user