diff options
35 files changed, 1098 insertions, 687 deletions
| diff --git a/src/android/app/src/main/java/org/yuzu/yuzu_emu/activities/EmulationActivity.kt b/src/android/app/src/main/java/org/yuzu/yuzu_emu/activities/EmulationActivity.kt index 93c8ce922..9b08f008d 100644 --- a/src/android/app/src/main/java/org/yuzu/yuzu_emu/activities/EmulationActivity.kt +++ b/src/android/app/src/main/java/org/yuzu/yuzu_emu/activities/EmulationActivity.kt @@ -49,7 +49,6 @@ import org.yuzu.yuzu_emu.utils.ForegroundService  import org.yuzu.yuzu_emu.utils.InputHandler  import org.yuzu.yuzu_emu.utils.Log  import org.yuzu.yuzu_emu.utils.MemoryUtil -import org.yuzu.yuzu_emu.utils.NativeConfig  import org.yuzu.yuzu_emu.utils.NfcReader  import org.yuzu.yuzu_emu.utils.ThemeHelper  import java.text.NumberFormat @@ -171,11 +170,6 @@ class EmulationActivity : AppCompatActivity(), SensorEventListener {          stopMotionSensorListener()      } -    override fun onStop() { -        super.onStop() -        NativeConfig.saveGlobalConfig() -    } -      override fun onUserLeaveHint() {          if (Build.VERSION.SDK_INT < Build.VERSION_CODES.S) {              if (BooleanSetting.PICTURE_IN_PICTURE.getBoolean() && !isInPictureInPictureMode) { diff --git a/src/android/app/src/main/java/org/yuzu/yuzu_emu/fragments/EmulationFragment.kt b/src/android/app/src/main/java/org/yuzu/yuzu_emu/fragments/EmulationFragment.kt index 9efc1705d..47767454a 100644 --- a/src/android/app/src/main/java/org/yuzu/yuzu_emu/fragments/EmulationFragment.kt +++ b/src/android/app/src/main/java/org/yuzu/yuzu_emu/fragments/EmulationFragment.kt @@ -554,6 +554,7 @@ class EmulationFragment : Fragment(), SurfaceHolder.Callback {              findItem(R.id.menu_touchscreen).isChecked = BooleanSetting.TOUCHSCREEN.getBoolean()          } +        popup.setOnDismissListener { NativeConfig.saveGlobalConfig() }          popup.setOnMenuItemClickListener {              when (it.itemId) {                  R.id.menu_toggle_fps -> { @@ -720,7 +721,9 @@ class EmulationFragment : Fragment(), SurfaceHolder.Callback {          MaterialAlertDialogBuilder(requireContext())              .setTitle(R.string.emulation_control_adjust)              .setView(adjustBinding.root) -            .setPositiveButton(android.R.string.ok, null) +            .setPositiveButton(android.R.string.ok) { _: DialogInterface?, _: Int -> +                NativeConfig.saveGlobalConfig() +            }              .setNeutralButton(R.string.slider_default) { _: DialogInterface?, _: Int ->                  setControlScale(50)                  setControlOpacity(100) diff --git a/src/core/CMakeLists.txt b/src/core/CMakeLists.txt index 293d9647b..16ddb5e90 100644 --- a/src/core/CMakeLists.txt +++ b/src/core/CMakeLists.txt @@ -712,22 +712,23 @@ add_library(core STATIC      hle/service/server_manager.h      hle/service/service.cpp      hle/service/service.h -    hle/service/set/appln_settings.cpp -    hle/service/set/appln_settings.h -    hle/service/set/device_settings.cpp -    hle/service/set/device_settings.h +    hle/service/set/setting_formats/appln_settings.cpp +    hle/service/set/setting_formats/appln_settings.h +    hle/service/set/setting_formats/device_settings.cpp +    hle/service/set/setting_formats/device_settings.h +    hle/service/set/setting_formats/system_settings.cpp +    hle/service/set/setting_formats/system_settings.h +    hle/service/set/setting_formats/private_settings.cpp +    hle/service/set/setting_formats/private_settings.h      hle/service/set/factory_settings_server.cpp      hle/service/set/factory_settings_server.h      hle/service/set/firmware_debug_settings_server.cpp      hle/service/set/firmware_debug_settings_server.h -    hle/service/set/private_settings.cpp -    hle/service/set/private_settings.h      hle/service/set/settings.cpp      hle/service/set/settings.h      hle/service/set/settings_server.cpp      hle/service/set/settings_server.h -    hle/service/set/system_settings.cpp -    hle/service/set/system_settings.h +    hle/service/set/settings_types.h      hle/service/set/system_settings_server.cpp      hle/service/set/system_settings_server.h      hle/service/sm/sm.cpp diff --git a/src/core/hle/service/acc/profile_manager.cpp b/src/core/hle/service/acc/profile_manager.cpp index 683f44e27..29a10ad13 100644 --- a/src/core/hle/service/acc/profile_manager.cpp +++ b/src/core/hle/service/acc/profile_manager.cpp @@ -11,6 +11,7 @@  #include "common/fs/path_util.h"  #include "common/polyfill_ranges.h"  #include "common/settings.h" +#include "common/string_util.h"  #include "core/hle/service/acc/profile_manager.h"  namespace Service::Account { @@ -164,6 +165,22 @@ std::optional<std::size_t> ProfileManager::GetUserIndex(const ProfileInfo& user)      return GetUserIndex(user.user_uuid);  } +/// Returns the first user profile seen based on username (which does not enforce uniqueness) +std::optional<std::size_t> ProfileManager::GetUserIndex(const std::string& username) const { +    const auto iter = +        std::find_if(profiles.begin(), profiles.end(), [&username](const ProfileInfo& p) { +            const std::string profile_username = Common::StringFromFixedZeroTerminatedBuffer( +                reinterpret_cast<const char*>(p.username.data()), p.username.size()); + +            return username.compare(profile_username) == 0; +        }); +    if (iter == profiles.end()) { +        return std::nullopt; +    } + +    return static_cast<std::size_t>(std::distance(profiles.begin(), iter)); +} +  /// Returns the data structure used by the switch when GetProfileBase is called on acc:*  bool ProfileManager::GetProfileBase(std::optional<std::size_t> index, ProfileBase& profile) const {      if (!index || index >= MAX_USERS) { diff --git a/src/core/hle/service/acc/profile_manager.h b/src/core/hle/service/acc/profile_manager.h index e21863e64..f94157300 100644 --- a/src/core/hle/service/acc/profile_manager.h +++ b/src/core/hle/service/acc/profile_manager.h @@ -70,6 +70,7 @@ public:      std::optional<Common::UUID> GetUser(std::size_t index) const;      std::optional<std::size_t> GetUserIndex(const Common::UUID& uuid) const;      std::optional<std::size_t> GetUserIndex(const ProfileInfo& user) const; +    std::optional<std::size_t> GetUserIndex(const std::string& username) const;      bool GetProfileBase(std::optional<std::size_t> index, ProfileBase& profile) const;      bool GetProfileBase(Common::UUID uuid, ProfileBase& profile) const;      bool GetProfileBase(const ProfileInfo& user, ProfileBase& profile) const; diff --git a/src/core/hle/service/hid/hid.cpp b/src/core/hle/service/hid/hid.cpp index 4ce0a9834..03ebdc137 100644 --- a/src/core/hle/service/hid/hid.cpp +++ b/src/core/hle/service/hid/hid.cpp @@ -20,7 +20,7 @@ void LoopProcess(Core::System& system) {      auto server_manager = std::make_unique<ServerManager>(system);      std::shared_ptr<ResourceManager> resource_manager = std::make_shared<ResourceManager>(system);      std::shared_ptr<HidFirmwareSettings> firmware_settings = -        std::make_shared<HidFirmwareSettings>(); +        std::make_shared<HidFirmwareSettings>(system);      // TODO: Remove this hack when am is emulated properly.      resource_manager->Initialize(); diff --git a/src/core/hle/service/nfc/common/device.cpp b/src/core/hle/service/nfc/common/device.cpp index 31cc87acc..cc7776efc 100644 --- a/src/core/hle/service/nfc/common/device.cpp +++ b/src/core/hle/service/nfc/common/device.cpp @@ -441,7 +441,10 @@ Result NfcDevice::Mount(NFP::ModelType model_type, NFP::MountTarget mount_target      device_state = DeviceState::TagMounted;      mount_target = mount_target_; -    if (!is_corrupted && mount_target != NFP::MountTarget::Rom) { +    const bool create_backup = +        mount_target == NFP::MountTarget::All || mount_target == NFP::MountTarget::Ram || +        (mount_target == NFP::MountTarget::Rom && HasBackup(encrypted_tag_data.uuid).IsError()); +    if (!is_corrupted && create_backup) {          std::vector<u8> data(sizeof(NFP::EncryptedNTAG215File));          memcpy(data.data(), &encrypted_tag_data, sizeof(encrypted_tag_data));          WriteBackupData(encrypted_tag_data.uuid, data); diff --git a/src/core/hle/service/set/private_settings.h b/src/core/hle/service/set/private_settings.h deleted file mode 100644 index b63eaf45c..000000000 --- a/src/core/hle/service/set/private_settings.h +++ /dev/null @@ -1,72 +0,0 @@ -// SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project -// SPDX-License-Identifier: GPL-2.0-or-later - -#pragma once - -#include <array> - -#include "common/bit_field.h" -#include "common/common_funcs.h" -#include "common/common_types.h" -#include "common/uuid.h" -#include "core/hle/service/time/clock_types.h" - -namespace Service::Set { - -/// This is nn::settings::system::InitialLaunchFlag -struct InitialLaunchFlag { -    union { -        u32 raw{}; - -        BitField<0, 1, u32> InitialLaunchCompletionFlag; -        BitField<8, 1, u32> InitialLaunchUserAdditionFlag; -        BitField<16, 1, u32> InitialLaunchTimestampFlag; -    }; -}; -static_assert(sizeof(InitialLaunchFlag) == 4, "InitialLaunchFlag is an invalid size"); - -/// This is nn::settings::system::InitialLaunchSettings -struct InitialLaunchSettings { -    InitialLaunchFlag flags; -    INSERT_PADDING_BYTES(0x4); -    Service::Time::Clock::SteadyClockTimePoint timestamp; -}; -static_assert(sizeof(InitialLaunchSettings) == 0x20, "InitialLaunchSettings is incorrect size"); - -#pragma pack(push, 4) -struct InitialLaunchSettingsPacked { -    InitialLaunchFlag flags; -    Service::Time::Clock::SteadyClockTimePoint timestamp; -}; -#pragma pack(pop) -static_assert(sizeof(InitialLaunchSettingsPacked) == 0x1C, -              "InitialLaunchSettingsPacked is incorrect size"); - -struct PrivateSettings { -    std::array<u8, 0x10> reserved_00; - -    // nn::settings::system::InitialLaunchSettings -    InitialLaunchSettings initial_launch_settings; - -    std::array<u8, 0x20> reserved_30; - -    Common::UUID external_clock_source_id; -    s64 shutdown_rtc_value; -    s64 external_steady_clock_internal_offset; - -    std::array<u8, 0x60> reserved_70; - -    // nn::settings::system::PlatformRegion -    std::array<u8, 0x4> platform_region; - -    std::array<u8, 0x4> reserved_D4; -}; -static_assert(offsetof(PrivateSettings, initial_launch_settings) == 0x10); -static_assert(offsetof(PrivateSettings, external_clock_source_id) == 0x50); -static_assert(offsetof(PrivateSettings, reserved_70) == 0x70); -static_assert(offsetof(PrivateSettings, platform_region) == 0xD0); -static_assert(sizeof(PrivateSettings) == 0xD8, "PrivateSettings has the wrong size!"); - -PrivateSettings DefaultPrivateSettings(); - -} // namespace Service::Set diff --git a/src/core/hle/service/set/appln_settings.cpp b/src/core/hle/service/set/setting_formats/appln_settings.cpp index a5d802757..f7c7d5b91 100644 --- a/src/core/hle/service/set/appln_settings.cpp +++ b/src/core/hle/service/set/setting_formats/appln_settings.cpp @@ -1,12 +1,16 @@  // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project  // SPDX-License-Identifier: GPL-2.0-or-later -#include "core/hle/service/set/appln_settings.h" +#include "core/hle/service/set/setting_formats/appln_settings.h"  namespace Service::Set {  ApplnSettings DefaultApplnSettings() { -    return {}; +    ApplnSettings settings{}; + +    settings.mii_author_id = Common::UUID::MakeDefault(); + +    return settings;  }  } // namespace Service::Set diff --git a/src/core/hle/service/set/appln_settings.h b/src/core/hle/service/set/setting_formats/appln_settings.h index 126375860..ba9af998a 100644 --- a/src/core/hle/service/set/appln_settings.h +++ b/src/core/hle/service/set/setting_formats/appln_settings.h @@ -7,24 +7,23 @@  #include <cstddef>  #include "common/common_types.h" +#include "common/uuid.h" +#include "core/hle/service/set/settings_types.h"  namespace Service::Set {  struct ApplnSettings { -    std::array<u8, 0x10> reserved_000; +    INSERT_PADDING_BYTES(0x10); // Reserved      // nn::util::Uuid MiiAuthorId, copied from system settings 0x94B0 -    std::array<u8, 0x10> mii_author_id; - -    std::array<u8, 0x30> reserved_020; +    Common::UUID mii_author_id; +    INSERT_PADDING_BYTES(0x30); // Reserved      // nn::settings::system::ServiceDiscoveryControlSettings -    std::array<u8, 0x4> service_discovery_control_settings; - -    std::array<u8, 0x20> reserved_054; +    u32 service_discovery_control_settings; +    INSERT_PADDING_BYTES(0x20); // Reserved      bool in_repair_process_enable_flag; - -    std::array<u8, 0x3> pad_075; +    INSERT_PADDING_BYTES(0x3);  };  static_assert(offsetof(ApplnSettings, mii_author_id) == 0x10);  static_assert(offsetof(ApplnSettings, service_discovery_control_settings) == 0x50); diff --git a/src/core/hle/service/set/device_settings.cpp b/src/core/hle/service/set/setting_formats/device_settings.cpp index e423ce38a..5f295404d 100644 --- a/src/core/hle/service/set/device_settings.cpp +++ b/src/core/hle/service/set/setting_formats/device_settings.cpp @@ -1,7 +1,7 @@  // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project  // SPDX-License-Identifier: GPL-2.0-or-later -#include "core/hle/service/set/device_settings.h" +#include "core/hle/service/set/setting_formats/device_settings.h"  namespace Service::Set { diff --git a/src/core/hle/service/set/device_settings.h b/src/core/hle/service/set/setting_formats/device_settings.h index f291d0ebe..2827756f6 100644 --- a/src/core/hle/service/set/device_settings.h +++ b/src/core/hle/service/set/setting_formats/device_settings.h @@ -7,10 +7,12 @@  #include <cstddef>  #include "common/common_types.h" +#include "common/vector_math.h" +#include "core/hle/service/set/settings_types.h"  namespace Service::Set {  struct DeviceSettings { -    std::array<u8, 0x10> reserved_000; +    INSERT_PADDING_BYTES(0x10); // Reserved      // nn::settings::BatteryLot      std::array<u8, 0x18> ptm_battery_lot; @@ -19,26 +21,24 @@ struct DeviceSettings {      u8 ptm_battery_version;      // nn::settings::system::PtmCycleCountReliability      u32 ptm_cycle_count_reliability; - -    std::array<u8, 0x48> reserved_048; +    INSERT_PADDING_BYTES(0x48); // Reserved      // nn::settings::system::AnalogStickUserCalibration L      std::array<u8, 0x10> analog_user_stick_calibration_l;      // nn::settings::system::AnalogStickUserCalibration R      std::array<u8, 0x10> analog_user_stick_calibration_r; - -    std::array<u8, 0x20> reserved_0B0; +    INSERT_PADDING_BYTES(0x20); // Reserved      // nn::settings::system::ConsoleSixAxisSensorAccelerationBias -    std::array<u8, 0xC> console_six_axis_sensor_acceleration_bias; +    Common::Vec3<f32> console_six_axis_sensor_acceleration_bias;      // nn::settings::system::ConsoleSixAxisSensorAngularVelocityBias -    std::array<u8, 0xC> console_six_axis_sensor_angular_velocity_bias; +    Common::Vec3<f32> console_six_axis_sensor_angular_velocity_bias;      // nn::settings::system::ConsoleSixAxisSensorAccelerationGain      std::array<u8, 0x24> console_six_axis_sensor_acceleration_gain;      // nn::settings::system::ConsoleSixAxisSensorAngularVelocityGain      std::array<u8, 0x24> console_six_axis_sensor_angular_velocity_gain;      // nn::settings::system::ConsoleSixAxisSensorAngularVelocityTimeBias -    std::array<u8, 0xC> console_six_axis_sensor_angular_velocity_time_bias; +    Common::Vec3<f32> console_six_axis_sensor_angular_velocity_time_bias;      // nn::settings::system::ConsoleSixAxisSensorAngularAcceleration      std::array<u8, 0x24> console_six_axis_sensor_angular_acceleration;  }; diff --git a/src/core/hle/service/set/private_settings.cpp b/src/core/hle/service/set/setting_formats/private_settings.cpp index 70bf65727..81c362482 100644 --- a/src/core/hle/service/set/private_settings.cpp +++ b/src/core/hle/service/set/setting_formats/private_settings.cpp @@ -1,7 +1,7 @@  // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project  // SPDX-License-Identifier: GPL-2.0-or-later -#include "core/hle/service/set/private_settings.h" +#include "core/hle/service/set/setting_formats/private_settings.h"  namespace Service::Set { diff --git a/src/core/hle/service/set/setting_formats/private_settings.h b/src/core/hle/service/set/setting_formats/private_settings.h new file mode 100644 index 000000000..6c40f62e1 --- /dev/null +++ b/src/core/hle/service/set/setting_formats/private_settings.h @@ -0,0 +1,39 @@ +// SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project +// SPDX-License-Identifier: GPL-2.0-or-later + +#pragma once + +#include <array> + +#include "common/common_types.h" +#include "common/uuid.h" +#include "core/hle/service/set/settings_types.h" +#include "core/hle/service/time/clock_types.h" + +namespace Service::Set { + +struct PrivateSettings { +    INSERT_PADDING_BYTES(0x10); // Reserved + +    InitialLaunchSettings initial_launch_settings; +    INSERT_PADDING_BYTES(0x20); // Reserved + +    Common::UUID external_clock_source_id; +    s64 shutdown_rtc_value; +    s64 external_steady_clock_internal_offset; +    INSERT_PADDING_BYTES(0x60); // Reserved + +    // nn::settings::system::PlatformRegion +    s32 platform_region; +    INSERT_PADDING_BYTES(0x4); // Reserved +}; +static_assert(offsetof(PrivateSettings, initial_launch_settings) == 0x10); +static_assert(offsetof(PrivateSettings, external_clock_source_id) == 0x50); +static_assert(offsetof(PrivateSettings, shutdown_rtc_value) == 0x60); +static_assert(offsetof(PrivateSettings, external_steady_clock_internal_offset) == 0x68); +static_assert(offsetof(PrivateSettings, platform_region) == 0xD0); +static_assert(sizeof(PrivateSettings) == 0xD8, "PrivateSettings has the wrong size!"); + +PrivateSettings DefaultPrivateSettings(); + +} // namespace Service::Set diff --git a/src/core/hle/service/set/system_settings.cpp b/src/core/hle/service/set/setting_formats/system_settings.cpp index 5977429b2..66e57651e 100644 --- a/src/core/hle/service/set/system_settings.cpp +++ b/src/core/hle/service/set/setting_formats/system_settings.cpp @@ -1,7 +1,7 @@  // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project  // SPDX-License-Identifier: GPL-2.0-or-later -#include "core/hle/service/set/system_settings.h" +#include "core/hle/service/set/setting_formats/system_settings.h"  namespace Service::Set { @@ -11,6 +11,8 @@ SystemSettings DefaultSystemSettings() {      settings.version = 0x140000;      settings.flags = 7; +    settings.mii_author_id = Common::UUID::MakeDefault(); +      settings.color_set_id = ColorSet::BasicWhite;      settings.notification_settings = { @@ -45,6 +47,10 @@ SystemSettings DefaultSystemSettings() {      settings.device_time_zone_location_name = {"UTC"};      settings.user_system_clock_automatic_correction_enabled = false; +    settings.primary_album_storage = PrimaryAlbumStorage::SdCard; +    settings.battery_percentage_flag = true; +    settings.chinese_traditional_input_method = ChineseTraditionalInputMethod::Unknown0; +      return settings;  } diff --git a/src/core/hle/service/set/system_settings.h b/src/core/hle/service/set/setting_formats/system_settings.h index 6ec9e71e7..14654f8b1 100644 --- a/src/core/hle/service/set/system_settings.h +++ b/src/core/hle/service/set/setting_formats/system_settings.h @@ -8,272 +8,14 @@  #include "common/bit_field.h"  #include "common/common_funcs.h"  #include "common/common_types.h" -#include "core/hle/service/set/private_settings.h" +#include "common/uuid.h" +#include "common/vector_math.h" +#include "core/hle/service/set/setting_formats/private_settings.h" +#include "core/hle/service/set/settings_types.h"  #include "core/hle/service/time/clock_types.h"  namespace Service::Set { -/// This is "nn::settings::LanguageCode", which is a NUL-terminated string stored in a u64. -enum class LanguageCode : u64 { -    JA = 0x000000000000616A, -    EN_US = 0x00000053552D6E65, -    FR = 0x0000000000007266, -    DE = 0x0000000000006564, -    IT = 0x0000000000007469, -    ES = 0x0000000000007365, -    ZH_CN = 0x0000004E432D687A, -    KO = 0x0000000000006F6B, -    NL = 0x0000000000006C6E, -    PT = 0x0000000000007470, -    RU = 0x0000000000007572, -    ZH_TW = 0x00000057542D687A, -    EN_GB = 0x00000042472D6E65, -    FR_CA = 0x00000041432D7266, -    ES_419 = 0x00003931342D7365, -    ZH_HANS = 0x00736E61482D687A, -    ZH_HANT = 0x00746E61482D687A, -    PT_BR = 0x00000052422D7470, -}; - -/// This is nn::settings::system::ErrorReportSharePermission -enum class ErrorReportSharePermission : u32 { -    NotConfirmed, -    Granted, -    Denied, -}; - -/// This is nn::settings::system::ChineseTraditionalInputMethod -enum class ChineseTraditionalInputMethod : u32 { -    Unknown0 = 0, -    Unknown1 = 1, -    Unknown2 = 2, -}; - -/// This is nn::settings::system::HomeMenuScheme -struct HomeMenuScheme { -    u32 main; -    u32 back; -    u32 sub; -    u32 bezel; -    u32 extra; -}; -static_assert(sizeof(HomeMenuScheme) == 0x14, "HomeMenuScheme is incorrect size"); - -/// Indicates the current theme set by the system settings -enum class ColorSet : u32 { -    BasicWhite = 0, -    BasicBlack = 1, -}; - -/// Indicates the current console is a retail or kiosk unit -enum class QuestFlag : u8 { -    Retail = 0, -    Kiosk = 1, -}; - -/// This is nn::settings::system::RegionCode -enum class RegionCode : u32 { -    Japan, -    Usa, -    Europe, -    Australia, -    HongKongTaiwanKorea, -    China, -}; - -/// This is nn::settings::system::AccountSettings -struct AccountSettings { -    u32 flags; -}; -static_assert(sizeof(AccountSettings) == 4, "AccountSettings is an invalid size"); - -/// This is nn::settings::system::NotificationVolume -enum class NotificationVolume : u32 { -    Mute, -    Low, -    High, -}; - -/// This is nn::settings::system::NotificationFlag -struct NotificationFlag { -    union { -        u32 raw{}; - -        BitField<0, 1, u32> RingtoneFlag; -        BitField<1, 1, u32> DownloadCompletionFlag; -        BitField<8, 1, u32> EnablesNews; -        BitField<9, 1, u32> IncomingLampFlag; -    }; -}; -static_assert(sizeof(NotificationFlag) == 4, "NotificationFlag is an invalid size"); - -/// This is nn::settings::system::NotificationTime -struct NotificationTime { -    u32 hour; -    u32 minute; -}; -static_assert(sizeof(NotificationTime) == 0x8, "NotificationTime is an invalid size"); - -/// This is nn::settings::system::NotificationSettings -struct NotificationSettings { -    NotificationFlag flags; -    NotificationVolume volume; -    NotificationTime start_time; -    NotificationTime stop_time; -}; -static_assert(sizeof(NotificationSettings) == 0x18, "NotificationSettings is an invalid size"); - -/// This is nn::settings::system::AccountNotificationFlag -struct AccountNotificationFlag { -    union { -        u32 raw{}; - -        BitField<0, 1, u32> FriendOnlineFlag; -        BitField<1, 1, u32> FriendRequestFlag; -        BitField<8, 1, u32> CoralInvitationFlag; -    }; -}; -static_assert(sizeof(AccountNotificationFlag) == 4, "AccountNotificationFlag is an invalid size"); - -/// This is nn::settings::system::FriendPresenceOverlayPermission -enum class FriendPresenceOverlayPermission : u8 { -    NotConfirmed, -    NoDisplay, -    FavoriteFriends, -    Friends, -}; - -/// This is nn::settings::system::AccountNotificationSettings -struct AccountNotificationSettings { -    Common::UUID uid; -    AccountNotificationFlag flags; -    FriendPresenceOverlayPermission friend_presence_permission; -    FriendPresenceOverlayPermission friend_invitation_permission; -    INSERT_PADDING_BYTES(0x2); -}; -static_assert(sizeof(AccountNotificationSettings) == 0x18, -              "AccountNotificationSettings is an invalid size"); - -/// This is nn::settings::system::TvFlag -struct TvFlag { -    union { -        u32 raw{}; - -        BitField<0, 1, u32> Allows4k; -        BitField<1, 1, u32> Allows3d; -        BitField<2, 1, u32> AllowsCec; -        BitField<3, 1, u32> PreventsScreenBurnIn; -    }; -}; -static_assert(sizeof(TvFlag) == 4, "TvFlag is an invalid size"); - -/// This is nn::settings::system::TvResolution -enum class TvResolution : u32 { -    Auto, -    Resolution1080p, -    Resolution720p, -    Resolution480p, -}; - -/// This is nn::settings::system::HdmiContentType -enum class HdmiContentType : u32 { -    None, -    Graphics, -    Cinema, -    Photo, -    Game, -}; - -/// This is nn::settings::system::RgbRange -enum class RgbRange : u32 { -    Auto, -    Full, -    Limited, -}; - -/// This is nn::settings::system::CmuMode -enum class CmuMode : u32 { -    None, -    ColorInvert, -    HighContrast, -    GrayScale, -}; - -/// This is nn::settings::system::TvSettings -struct TvSettings { -    TvFlag flags; -    TvResolution tv_resolution; -    HdmiContentType hdmi_content_type; -    RgbRange rgb_range; -    CmuMode cmu_mode; -    u32 tv_underscan; -    f32 tv_gama; -    f32 contrast_ratio; -}; -static_assert(sizeof(TvSettings) == 0x20, "TvSettings is an invalid size"); - -/// This is nn::settings::system::PrimaryAlbumStorage -enum class PrimaryAlbumStorage : u32 { -    Nand, -    SdCard, -}; - -/// This is nn::settings::system::HandheldSleepPlan -enum class HandheldSleepPlan : u32 { -    Sleep1Min, -    Sleep3Min, -    Sleep5Min, -    Sleep10Min, -    Sleep30Min, -    Never, -}; - -/// This is nn::settings::system::ConsoleSleepPlan -enum class ConsoleSleepPlan : u32 { -    Sleep1Hour, -    Sleep2Hour, -    Sleep3Hour, -    Sleep6Hour, -    Sleep12Hour, -    Never, -}; - -/// This is nn::settings::system::SleepFlag -struct SleepFlag { -    union { -        u32 raw{}; - -        BitField<0, 1, u32> SleepsWhilePlayingMedia; -        BitField<1, 1, u32> WakesAtPowerStateChange; -    }; -}; -static_assert(sizeof(SleepFlag) == 4, "TvFlag is an invalid size"); - -/// This is nn::settings::system::SleepSettings -struct SleepSettings { -    SleepFlag flags; -    HandheldSleepPlan handheld_sleep_plan; -    ConsoleSleepPlan console_sleep_plan; -}; -static_assert(sizeof(SleepSettings) == 0xc, "SleepSettings is incorrect size"); - -/// This is nn::settings::system::EulaVersionClockType -enum class EulaVersionClockType : u32 { -    NetworkSystemClock, -    SteadyClock, -}; - -/// This is nn::settings::system::EulaVersion -struct EulaVersion { -    u32 version; -    RegionCode region_code; -    EulaVersionClockType clock_type; -    INSERT_PADDING_BYTES(0x4); -    s64 posix_time; -    Time::Clock::SteadyClockTimePoint timestamp; -}; -static_assert(sizeof(EulaVersion) == 0x30, "EulaVersion is incorrect size"); -  struct SystemSettings {      // 0/unwritten (1.0.0), 0x20000 (2.0.0), 0x30000 (3.0.0-3.0.1), 0x40001 (4.0.0-4.1.0), 0x50000      // (5.0.0-5.1.0), 0x60000 (6.0.0-6.2.0), 0x70000 (7.0.0), 0x80000 (8.0.0-8.1.1), 0x90000 @@ -283,20 +25,16 @@ struct SystemSettings {      // 0/unwritten (1.0.0), 1 (6.0.0-8.1.0), 2 (8.1.1), 7 (9.0.0+).      // if (flags & 2), defaults are written for AnalogStickUserCalibration      u32 flags; +    INSERT_PADDING_BYTES(0x8); // Reserved -    std::array<u8, 0x8> reserved_00008; - -    // nn::settings::LanguageCode      LanguageCode language_code; - -    std::array<u8, 0x38> reserved_00018; +    INSERT_PADDING_BYTES(0x38); // Reserved      // nn::settings::system::NetworkSettings      u32 network_setting_count;      bool wireless_lan_enable_flag; -    std::array<u8, 0x3> pad_00055; - -    std::array<u8, 0x8> reserved_00058; +    INSERT_PADDING_BYTES(0x3); +    INSERT_PADDING_BYTES(0x8); // Reserved      // nn::settings::system::NetworkSettings      std::array<std::array<u8, 0x400>, 32> network_settings_1B0; @@ -304,161 +42,142 @@ struct SystemSettings {      // nn::settings::system::BluetoothDevicesSettings      std::array<u8, 0x4> bluetooth_device_settings_count;      bool bluetooth_enable_flag; -    std::array<u8, 0x3> pad_08065; +    INSERT_PADDING_BYTES(0x3);      bool bluetooth_afh_enable_flag; -    std::array<u8, 0x3> pad_08069; +    INSERT_PADDING_BYTES(0x3);      bool bluetooth_boost_enable_flag; -    std::array<u8, 0x3> pad_0806D; +    INSERT_PADDING_BYTES(0x3);      std::array<std::array<u8, 0x200>, 10> bluetooth_device_settings_first_10;      s32 ldn_channel; - -    std::array<u8, 0x3C> reserved_09474; +    INSERT_PADDING_BYTES(0x3C); // Reserved      // nn::util::Uuid MiiAuthorId -    std::array<u8, 0x10> mii_author_id; +    Common::UUID mii_author_id; -    std::array<u8, 0x30> reserved_094C0; +    INSERT_PADDING_BYTES(0x30); // Reserved      // nn::settings::system::NxControllerSettings      u32 nx_controller_settings_count; -    std::array<u8, 0xC> reserved_094F4; +    INSERT_PADDING_BYTES(0xC); // Reserved      // nn::settings::system::NxControllerSettings,      // nn::settings::system::NxControllerLegacySettings on 13.0.0+      std::array<std::array<u8, 0x40>, 10> nx_controller_legacy_settings; - -    std::array<u8, 0x170> reserved_09780; +    INSERT_PADDING_BYTES(0x170); // Reserved      bool external_rtc_reset_flag; -    std::array<u8, 0x3> pad_098F1; - -    std::array<u8, 0x3C> reserved_098F4; +    INSERT_PADDING_BYTES(0x3); +    INSERT_PADDING_BYTES(0x3C); // Reserved      s32 push_notification_activity_mode_on_sleep; +    INSERT_PADDING_BYTES(0x3C); // Reserved -    std::array<u8, 0x3C> reserved_09934; - -    // nn::settings::system::ErrorReportSharePermission      ErrorReportSharePermission error_report_share_permission; +    INSERT_PADDING_BYTES(0x3C); // Reserved -    std::array<u8, 0x3C> reserved_09974; - -    // nn::settings::KeyboardLayout -    std::array<u8, 0x4> keyboard_layout; - -    std::array<u8, 0x3C> reserved_099B4; +    KeyboardLayout keyboard_layout; +    INSERT_PADDING_BYTES(0x3C); // Reserved      bool web_inspector_flag; -    std::array<u8, 0x3> pad_099F1; +    INSERT_PADDING_BYTES(0x3);      // nn::settings::system::AllowedSslHost      u32 allowed_ssl_host_count;      bool memory_usage_rate_flag; -    std::array<u8, 0x3> pad_099F9; - -    std::array<u8, 0x34> reserved_099FC; +    INSERT_PADDING_BYTES(0x3); +    INSERT_PADDING_BYTES(0x34); // Reserved      // nn::settings::system::HostFsMountPoint      std::array<u8, 0x100> host_fs_mount_point;      // nn::settings::system::AllowedSslHost      std::array<std::array<u8, 0x100>, 8> allowed_ssl_hosts; - -    std::array<u8, 0x6C0> reserved_0A330; +    INSERT_PADDING_BYTES(0x6C0); // Reserved      // nn::settings::system::BlePairingSettings      u32 ble_pairing_settings_count; -    std::array<u8, 0xC> reserved_0A9F4; +    INSERT_PADDING_BYTES(0xC); // Reserved      std::array<std::array<u8, 0x80>, 10> ble_pairing_settings;      // nn::settings::system::AccountOnlineStorageSettings      u32 account_online_storage_settings_count; -    std::array<u8, 0xC> reserved_0AF04; +    INSERT_PADDING_BYTES(0xC); // Reserved      std::array<std::array<u8, 0x40>, 8> account_online_storage_settings;      bool pctl_ready_flag; -    std::array<u8, 0x3> pad_0B111; - -    std::array<u8, 0x3C> reserved_0B114; +    INSERT_PADDING_BYTES(0x3); +    INSERT_PADDING_BYTES(0x3C); // Reserved      // nn::settings::system::ThemeId      std::array<u8, 0x80> theme_id_type0;      std::array<u8, 0x80> theme_id_type1; +    INSERT_PADDING_BYTES(0x100); // Reserved -    std::array<u8, 0x100> reserved_0B250; - -    // nn::settings::ChineseTraditionalInputMethod      ChineseTraditionalInputMethod chinese_traditional_input_method; - -    std::array<u8, 0x3C> reserved_0B354; +    INSERT_PADDING_BYTES(0x3C); // Reserved      bool zoom_flag; -    std::array<u8, 0x3> pad_0B391; - -    std::array<u8, 0x3C> reserved_0B394; +    INSERT_PADDING_BYTES(0x3); +    INSERT_PADDING_BYTES(0x3C); // Reserved      // nn::settings::system::ButtonConfigRegisteredSettings      u32 button_config_registered_settings_count; -    std::array<u8, 0xC> reserved_0B3D4; +    INSERT_PADDING_BYTES(0xC); // Reserved      // nn::settings::system::ButtonConfigSettings      u32 button_config_settings_count; -    std::array<u8, 0x4> reserved_0B3E4; +    INSERT_PADDING_BYTES(0x4); // Reserved      std::array<std::array<u8, 0x5A8>, 5> button_config_settings; -    std::array<u8, 0x13B0> reserved_0D030; +    INSERT_PADDING_BYTES(0x13B0); // Reserved      u32 button_config_settings_embedded_count; -    std::array<u8, 0x4> reserved_0E3E4; +    INSERT_PADDING_BYTES(0x4); // Reserved      std::array<std::array<u8, 0x5A8>, 5> button_config_settings_embedded; -    std::array<u8, 0x13B0> reserved_10030; +    INSERT_PADDING_BYTES(0x13B0); // Reserved      u32 button_config_settings_left_count; -    std::array<u8, 0x4> reserved_113E4; +    INSERT_PADDING_BYTES(0x4); // Reserved      std::array<std::array<u8, 0x5A8>, 5> button_config_settings_left; -    std::array<u8, 0x13B0> reserved_13030; +    INSERT_PADDING_BYTES(0x13B0); // Reserved      u32 button_config_settings_right_count; -    std::array<u8, 0x4> reserved_143E4; +    INSERT_PADDING_BYTES(0x4); // Reserved      std::array<std::array<u8, 0x5A8>, 5> button_config_settings_right; -    std::array<u8, 0x73B0> reserved_16030; +    INSERT_PADDING_BYTES(0x73B0); // Reserved      // nn::settings::system::ButtonConfigRegisteredSettings      std::array<u8, 0x5C8> button_config_registered_settings_embedded;      std::array<std::array<u8, 0x5C8>, 10> button_config_registered_settings; - -    std::array<u8, 0x7FF8> reserved_21378; +    INSERT_PADDING_BYTES(0x7FF8); // Reserved      // nn::settings::system::ConsoleSixAxisSensorAccelerationBias -    std::array<u8, 0xC> console_six_axis_sensor_acceleration_bias; +    Common::Vec3<f32> console_six_axis_sensor_acceleration_bias;      // nn::settings::system::ConsoleSixAxisSensorAngularVelocityBias -    std::array<u8, 0xC> console_six_axis_sensor_angular_velocity_bias; +    Common::Vec3<f32> console_six_axis_sensor_angular_velocity_bias;      // nn::settings::system::ConsoleSixAxisSensorAccelerationGain      std::array<u8, 0x24> console_six_axis_sensor_acceleration_gain;      // nn::settings::system::ConsoleSixAxisSensorAngularVelocityGain      std::array<u8, 0x24> console_six_axis_sensor_angular_velocity_gain;      // nn::settings::system::ConsoleSixAxisSensorAngularVelocityTimeBias -    std::array<u8, 0xC> console_six_axis_sensor_angular_velocity_time_bias; +    Common::Vec3<f32> console_six_axis_sensor_angular_velocity_time_bias;      // nn::settings::system::ConsoleSixAxisSensorAngularAcceleration      std::array<u8, 0x24> console_six_axis_sensor_angular_velocity_acceleration; - -    std::array<u8, 0x70> reserved_29400; +    INSERT_PADDING_BYTES(0x70); // Reserved      bool lock_screen_flag; -    std::array<u8, 0x3> pad_29471; - -    std::array<u8, 0x4> reserved_249274; +    INSERT_PADDING_BYTES(0x3); +    INSERT_PADDING_BYTES(0x4); // Reserved      ColorSet color_set_id;      QuestFlag quest_flag; -    // nn::settings::system::RegionCode -    RegionCode region_code; +    SystemRegionCode region_code;      // Different to nn::settings::system::InitialLaunchSettings?      InitialLaunchSettingsPacked initial_launch_settings_packed;      bool battery_percentage_flag; -    std::array<u8, 0x3> pad_294A1; +    INSERT_PADDING_BYTES(0x3);      // BitFlagSet<32, nn::settings::system::AppletLaunchFlag>      u32 applet_launch_flag; @@ -469,33 +188,26 @@ struct SystemSettings {      std::array<u8, 0x10> theme_key;      bool field_testing_flag; -    std::array<u8, 0x3> pad_294C1; +    INSERT_PADDING_BYTES(0x3);      s32 panel_crc_mode; - -    std::array<u8, 0x28> reserved_294C8; +    INSERT_PADDING_BYTES(0x28); // Reserved      // nn::settings::system::BacklightSettings      std::array<u8, 0x2C> backlight_settings_mixed_up; +    INSERT_PADDING_BYTES(0x64); // Reserved -    std::array<u8, 0x64> reserved_2951C; - -    // nn::time::SystemClockContext      Service::Time::Clock::SystemClockContext user_system_clock_context;      Service::Time::Clock::SystemClockContext network_system_clock_context;      bool user_system_clock_automatic_correction_enabled; -    std::array<u8, 0x3> pad_295C1; -    std::array<u8, 0x4> reserved_295C4; -    // nn::time::SteadyClockTimePoint +    INSERT_PADDING_BYTES(0x3); +    INSERT_PADDING_BYTES(0x4); // Reserved      Service::Time::Clock::SteadyClockTimePoint          user_system_clock_automatic_correction_updated_time_point; +    INSERT_PADDING_BYTES(0x10); // Reserved -    std::array<u8, 0x10> reserved_295E0; - -    // nn::settings::system::AccountSettings      AccountSettings account_settings; - -    std::array<u8, 0xFC> reserved_295F4; +    INSERT_PADDING_BYTES(0xFC); // Reserved      // nn::settings::system::AudioVolume      std::array<u8, 0x8> audio_volume_type0; @@ -505,47 +217,42 @@ struct SystemSettings {      s32 audio_output_mode_type1;      s32 audio_output_mode_type2;      bool force_mute_on_headphone_removed; -    std::array<u8, 0x3> pad_2970D; +    INSERT_PADDING_BYTES(0x3);      s32 headphone_volume_warning_count;      bool heaphone_volume_update_flag; -    std::array<u8, 0x3> pad_29715; +    INSERT_PADDING_BYTES(0x3);      // nn::settings::system::AudioVolume      std::array<u8, 0x8> audio_volume_type2;      // nn::settings::system::AudioOutputMode      s32 audio_output_mode_type3;      s32 audio_output_mode_type4;      bool hearing_protection_safeguard_flag; -    std::array<u8, 0x3> pad_29729; -    std::array<u8, 0x4> reserved_2972C; +    INSERT_PADDING_BYTES(0x3); +    INSERT_PADDING_BYTES(0x4); // Reserved      s64 hearing_protection_safeguard_remaining_time; -    std::array<u8, 0x38> reserved_29738; +    INSERT_PADDING_BYTES(0x38); // Reserved      bool console_information_upload_flag; -    std::array<u8, 0x3> pad_29771; - -    std::array<u8, 0x3C> reserved_29774; +    INSERT_PADDING_BYTES(0x3); +    INSERT_PADDING_BYTES(0x3C); // Reserved      bool automatic_application_download_flag; -    std::array<u8, 0x3> pad_297B1; - -    std::array<u8, 0x4> reserved_297B4; +    INSERT_PADDING_BYTES(0x3); +    INSERT_PADDING_BYTES(0x4); // Reserved -    // nn::settings::system::NotificationSettings      NotificationSettings notification_settings; - -    std::array<u8, 0x60> reserved_297D0; +    INSERT_PADDING_BYTES(0x60); // Reserved      // nn::settings::system::AccountNotificationSettings      u32 account_notification_settings_count; -    std::array<u8, 0xC> reserved_29834; +    INSERT_PADDING_BYTES(0xC); // Reserved      std::array<AccountNotificationSettings, 8> account_notification_settings; - -    std::array<u8, 0x140> reserved_29900; +    INSERT_PADDING_BYTES(0x140); // Reserved      f32 vibration_master_volume;      bool usb_full_key_enable_flag; -    std::array<u8, 0x3> pad_29A45; +    INSERT_PADDING_BYTES(0x3);      // nn::settings::system::AnalogStickUserCalibration      std::array<u8, 0x10> analog_stick_user_calibration_left; @@ -553,85 +260,68 @@ struct SystemSettings {      // nn::settings::system::TouchScreenMode      s32 touch_screen_mode; +    INSERT_PADDING_BYTES(0x14); // Reserved -    std::array<u8, 0x14> reserved_29A6C; - -    // nn::settings::system::TvSettings      TvSettings tv_settings;      // nn::settings::system::Edid      std::array<u8, 0x100> edid; - -    std::array<u8, 0x2E0> reserved_29BA0; +    INSERT_PADDING_BYTES(0x2E0); // Reserved      // nn::settings::system::DataDeletionSettings      std::array<u8, 0x8> data_deletion_settings; - -    std::array<u8, 0x38> reserved_29E88; +    INSERT_PADDING_BYTES(0x38); // Reserved      // nn::ncm::ProgramId      std::array<u8, 0x8> initial_system_applet_program_id;      std::array<u8, 0x8> overlay_disp_program_id; - -    std::array<u8, 0x4> reserved_29ED0; +    INSERT_PADDING_BYTES(0x4); // Reserved      bool requires_run_repair_time_reviser; +    INSERT_PADDING_BYTES(0x6B); // Reserved -    std::array<u8, 0x6B> reserved_29ED5; - -    // nn::time::LocationName      Service::Time::TimeZone::LocationName device_time_zone_location_name; -    std::array<u8, 0x4> reserved_29F64; -    // nn::time::SteadyClockTimePoint +    INSERT_PADDING_BYTES(0x4); // Reserved      Service::Time::Clock::SteadyClockTimePoint device_time_zone_location_updated_time; - -    std::array<u8, 0xC0> reserved_29F80; +    INSERT_PADDING_BYTES(0xC0); // Reserved      // nn::settings::system::PrimaryAlbumStorage      PrimaryAlbumStorage primary_album_storage; - -    std::array<u8, 0x3C> reserved_2A044; +    INSERT_PADDING_BYTES(0x3C); // Reserved      bool usb_30_enable_flag; -    std::array<u8, 0x3> pad_2A081; +    INSERT_PADDING_BYTES(0x3);      bool usb_30_host_enable_flag; -    std::array<u8, 0x3> pad_2A085; +    INSERT_PADDING_BYTES(0x3);      bool usb_30_device_enable_flag; -    std::array<u8, 0x3> pad_2A089; - -    std::array<u8, 0x34> reserved_2A08C; +    INSERT_PADDING_BYTES(0x3); +    INSERT_PADDING_BYTES(0x34); // Reserved      bool nfc_enable_flag; -    std::array<u8, 0x3> pad_2A0C1; - -    std::array<u8, 0x3C> reserved_2A0C4; +    INSERT_PADDING_BYTES(0x3); +    INSERT_PADDING_BYTES(0x3C); // Reserved      // nn::settings::system::SleepSettings      SleepSettings sleep_settings; - -    std::array<u8, 0x34> reserved_2A10C; +    INSERT_PADDING_BYTES(0x34); // Reserved      // nn::settings::system::EulaVersion      u32 eula_version_count; -    std::array<u8, 0xC> reserved_2A144; +    INSERT_PADDING_BYTES(0xC); // Reserved      std::array<EulaVersion, 32> eula_versions; - -    std::array<u8, 0x200> reserved_2A750; +    INSERT_PADDING_BYTES(0x200); // Reserved      // nn::settings::system::DeviceNickName      std::array<u8, 0x80> device_nick_name; - -    std::array<u8, 0x80> reserved_2A9D0; +    INSERT_PADDING_BYTES(0x80); // Reserved      bool auto_update_enable_flag; -    std::array<u8, 0x3> pad_2AA51; - -    std::array<u8, 0x4C> reserved_2AA54; +    INSERT_PADDING_BYTES(0x3); +    INSERT_PADDING_BYTES(0x4C); // Reserved      // nn::settings::system::BluetoothDevicesSettings      std::array<std::array<u8, 0x200>, 14> bluetooth_device_settings_last_14; - -    std::array<u8, 0x2000> reserved_2C6A0; +    INSERT_PADDING_BYTES(0x2000); // Reserved      // nn::settings::system::NxControllerSettings      std::array<std::array<u8, 0x800>, 10> nx_controller_settings_data_from_offset_30; diff --git a/src/core/hle/service/set/settings_server.h b/src/core/hle/service/set/settings_server.h index a4e78db6c..8304e8424 100644 --- a/src/core/hle/service/set/settings_server.h +++ b/src/core/hle/service/set/settings_server.h @@ -4,72 +4,13 @@  #pragma once  #include "core/hle/service/service.h" -#include "core/hle/service/set/system_settings.h" +#include "core/hle/service/set/settings_types.h"  namespace Core {  class System;  }  namespace Service::Set { -enum class KeyboardLayout : u64 { -    Japanese = 0, -    EnglishUs = 1, -    EnglishUsInternational = 2, -    EnglishUk = 3, -    French = 4, -    FrenchCa = 5, -    Spanish = 6, -    SpanishLatin = 7, -    German = 8, -    Italian = 9, -    Portuguese = 10, -    Russian = 11, -    Korean = 12, -    ChineseSimplified = 13, -    ChineseTraditional = 14, -}; - -constexpr std::array<LanguageCode, 18> available_language_codes = {{ -    LanguageCode::JA, -    LanguageCode::EN_US, -    LanguageCode::FR, -    LanguageCode::DE, -    LanguageCode::IT, -    LanguageCode::ES, -    LanguageCode::ZH_CN, -    LanguageCode::KO, -    LanguageCode::NL, -    LanguageCode::PT, -    LanguageCode::RU, -    LanguageCode::ZH_TW, -    LanguageCode::EN_GB, -    LanguageCode::FR_CA, -    LanguageCode::ES_419, -    LanguageCode::ZH_HANS, -    LanguageCode::ZH_HANT, -    LanguageCode::PT_BR, -}}; - -static constexpr std::array<std::pair<LanguageCode, KeyboardLayout>, 18> language_to_layout{{ -    {LanguageCode::JA, KeyboardLayout::Japanese}, -    {LanguageCode::EN_US, KeyboardLayout::EnglishUs}, -    {LanguageCode::FR, KeyboardLayout::French}, -    {LanguageCode::DE, KeyboardLayout::German}, -    {LanguageCode::IT, KeyboardLayout::Italian}, -    {LanguageCode::ES, KeyboardLayout::Spanish}, -    {LanguageCode::ZH_CN, KeyboardLayout::ChineseSimplified}, -    {LanguageCode::KO, KeyboardLayout::Korean}, -    {LanguageCode::NL, KeyboardLayout::EnglishUsInternational}, -    {LanguageCode::PT, KeyboardLayout::Portuguese}, -    {LanguageCode::RU, KeyboardLayout::Russian}, -    {LanguageCode::ZH_TW, KeyboardLayout::ChineseTraditional}, -    {LanguageCode::EN_GB, KeyboardLayout::EnglishUk}, -    {LanguageCode::FR_CA, KeyboardLayout::FrenchCa}, -    {LanguageCode::ES_419, KeyboardLayout::SpanishLatin}, -    {LanguageCode::ZH_HANS, KeyboardLayout::ChineseSimplified}, -    {LanguageCode::ZH_HANT, KeyboardLayout::ChineseTraditional}, -    {LanguageCode::PT_BR, KeyboardLayout::Portuguese}, -}};  LanguageCode GetLanguageCodeFromIndex(std::size_t idx); diff --git a/src/core/hle/service/set/settings_types.h b/src/core/hle/service/set/settings_types.h new file mode 100644 index 000000000..4dee202d7 --- /dev/null +++ b/src/core/hle/service/set/settings_types.h @@ -0,0 +1,451 @@ +// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project +// SPDX-License-Identifier: GPL-3.0-or-later + +#pragma once + +#include <array> + +#include "common/bit_field.h" +#include "common/common_funcs.h" +#include "common/common_types.h" +#include "common/uuid.h" +#include "core/hle/service/time/clock_types.h" + +namespace Service::Set { + +/// This is nn::settings::system::AudioOutputMode +enum class AudioOutputMode : u32 { +    ch_1, +    ch_2, +    ch_5_1, +    ch_7_1, +}; + +/// This is nn::settings::system::AudioOutputModeTarget +enum class AudioOutputModeTarget : u32 { +    Hdmi, +    Speaker, +    Headphone, +}; + +/// This is nn::settings::system::AudioVolumeTarget +enum class AudioVolumeTarget : u32 { +    Speaker, +    Headphone, +}; + +/// This is nn::settings::system::ClockSourceId +enum class ClockSourceId : u32 { +    NetworkSystemClock, +    SteadyClock, +}; + +/// This is nn::settings::system::CmuMode +enum class CmuMode : u32 { +    None, +    ColorInvert, +    HighContrast, +    GrayScale, +}; + +/// This is nn::settings::system::ChineseTraditionalInputMethod +enum class ChineseTraditionalInputMethod : u32 { +    Unknown0 = 0, +    Unknown1 = 1, +    Unknown2 = 2, +}; + +/// Indicates the current theme set by the system settings +enum class ColorSet : u32 { +    BasicWhite = 0, +    BasicBlack = 1, +}; + +/// This is nn::settings::system::ConsoleSleepPlan +enum class ConsoleSleepPlan : u32 { +    Sleep1Hour, +    Sleep2Hour, +    Sleep3Hour, +    Sleep6Hour, +    Sleep12Hour, +    Never, +}; + +/// This is nn::settings::system::ErrorReportSharePermission +enum class ErrorReportSharePermission : u32 { +    NotConfirmed, +    Granted, +    Denied, +}; + +/// This is nn::settings::system::EulaVersionClockType +enum class EulaVersionClockType : u32 { +    NetworkSystemClock, +    SteadyClock, +}; + +/// This is nn::settings::factory::RegionCode +enum class FactoryRegionCode : u32 { +    Japan, +    Usa, +    Europe, +    Australia, +    China, +    Korea, +    Taiwan, +}; + +/// This is nn::settings::system::FriendPresenceOverlayPermission +enum class FriendPresenceOverlayPermission : u8 { +    NotConfirmed, +    NoDisplay, +    FavoriteFriends, +    Friends, +}; + +enum class GetFirmwareVersionType { +    Version1, +    Version2, +}; + +/// This is nn::settings::system::HandheldSleepPlan +enum class HandheldSleepPlan : u32 { +    Sleep1Min, +    Sleep3Min, +    Sleep5Min, +    Sleep10Min, +    Sleep30Min, +    Never, +}; + +/// This is nn::settings::system::HdmiContentType +enum class HdmiContentType : u32 { +    None, +    Graphics, +    Cinema, +    Photo, +    Game, +}; + +enum class KeyboardLayout : u32 { +    Japanese = 0, +    EnglishUs = 1, +    EnglishUsInternational = 2, +    EnglishUk = 3, +    French = 4, +    FrenchCa = 5, +    Spanish = 6, +    SpanishLatin = 7, +    German = 8, +    Italian = 9, +    Portuguese = 10, +    Russian = 11, +    Korean = 12, +    ChineseSimplified = 13, +    ChineseTraditional = 14, +}; + +/// This is "nn::settings::LanguageCode", which is a NUL-terminated string stored in a u64. +enum class LanguageCode : u64 { +    JA = 0x000000000000616A, +    EN_US = 0x00000053552D6E65, +    FR = 0x0000000000007266, +    DE = 0x0000000000006564, +    IT = 0x0000000000007469, +    ES = 0x0000000000007365, +    ZH_CN = 0x0000004E432D687A, +    KO = 0x0000000000006F6B, +    NL = 0x0000000000006C6E, +    PT = 0x0000000000007470, +    RU = 0x0000000000007572, +    ZH_TW = 0x00000057542D687A, +    EN_GB = 0x00000042472D6E65, +    FR_CA = 0x00000041432D7266, +    ES_419 = 0x00003931342D7365, +    ZH_HANS = 0x00736E61482D687A, +    ZH_HANT = 0x00746E61482D687A, +    PT_BR = 0x00000052422D7470, +}; + +/// This is nn::settings::system::NotificationVolume +enum class NotificationVolume : u32 { +    Mute, +    Low, +    High, +}; + +/// This is nn::settings::system::PrimaryAlbumStorage +enum class PrimaryAlbumStorage : u32 { +    Nand, +    SdCard, +}; + +/// Indicates the current console is a retail or kiosk unit +enum class QuestFlag : u8 { +    Retail = 0, +    Kiosk = 1, +}; + +/// This is nn::settings::system::RgbRange +enum class RgbRange : u32 { +    Auto, +    Full, +    Limited, +}; + +/// This is nn::settings::system::RegionCode +enum class SystemRegionCode : u32 { +    Japan, +    Usa, +    Europe, +    Australia, +    HongKongTaiwanKorea, +    China, +}; + +/// This is nn::settings::system::TouchScreenMode +enum class TouchScreenMode : u32 { +    Stylus, +    Standard, +}; + +/// This is nn::settings::system::TvResolution +enum class TvResolution : u32 { +    Auto, +    Resolution1080p, +    Resolution720p, +    Resolution480p, +}; + +constexpr std::array<LanguageCode, 18> available_language_codes = {{ +    LanguageCode::JA, +    LanguageCode::EN_US, +    LanguageCode::FR, +    LanguageCode::DE, +    LanguageCode::IT, +    LanguageCode::ES, +    LanguageCode::ZH_CN, +    LanguageCode::KO, +    LanguageCode::NL, +    LanguageCode::PT, +    LanguageCode::RU, +    LanguageCode::ZH_TW, +    LanguageCode::EN_GB, +    LanguageCode::FR_CA, +    LanguageCode::ES_419, +    LanguageCode::ZH_HANS, +    LanguageCode::ZH_HANT, +    LanguageCode::PT_BR, +}}; + +static constexpr std::array<std::pair<LanguageCode, KeyboardLayout>, 18> language_to_layout{{ +    {LanguageCode::JA, KeyboardLayout::Japanese}, +    {LanguageCode::EN_US, KeyboardLayout::EnglishUs}, +    {LanguageCode::FR, KeyboardLayout::French}, +    {LanguageCode::DE, KeyboardLayout::German}, +    {LanguageCode::IT, KeyboardLayout::Italian}, +    {LanguageCode::ES, KeyboardLayout::Spanish}, +    {LanguageCode::ZH_CN, KeyboardLayout::ChineseSimplified}, +    {LanguageCode::KO, KeyboardLayout::Korean}, +    {LanguageCode::NL, KeyboardLayout::EnglishUsInternational}, +    {LanguageCode::PT, KeyboardLayout::Portuguese}, +    {LanguageCode::RU, KeyboardLayout::Russian}, +    {LanguageCode::ZH_TW, KeyboardLayout::ChineseTraditional}, +    {LanguageCode::EN_GB, KeyboardLayout::EnglishUk}, +    {LanguageCode::FR_CA, KeyboardLayout::FrenchCa}, +    {LanguageCode::ES_419, KeyboardLayout::SpanishLatin}, +    {LanguageCode::ZH_HANS, KeyboardLayout::ChineseSimplified}, +    {LanguageCode::ZH_HANT, KeyboardLayout::ChineseTraditional}, +    {LanguageCode::PT_BR, KeyboardLayout::Portuguese}, +}}; + +/// This is nn::settings::system::AccountNotificationFlag +struct AccountNotificationFlag { +    union { +        u32 raw{}; + +        BitField<0, 1, u32> FriendOnlineFlag; +        BitField<1, 1, u32> FriendRequestFlag; +        BitField<8, 1, u32> CoralInvitationFlag; +    }; +}; +static_assert(sizeof(AccountNotificationFlag) == 4, "AccountNotificationFlag is an invalid size"); + +/// This is nn::settings::system::AccountSettings +struct AccountSettings { +    u32 flags; +}; +static_assert(sizeof(AccountSettings) == 4, "AccountSettings is an invalid size"); + +/// This is nn::settings::system::DataDeletionFlag +struct DataDeletionFlag { +    union { +        u32 raw{}; + +        BitField<0, 1, u32> AutomaticDeletionFlag; +    }; +}; +static_assert(sizeof(DataDeletionFlag) == 4, "DataDeletionFlag is an invalid size"); + +/// This is nn::settings::system::InitialLaunchFlag +struct InitialLaunchFlag { +    union { +        u32 raw{}; + +        BitField<0, 1, u32> InitialLaunchCompletionFlag; +        BitField<8, 1, u32> InitialLaunchUserAdditionFlag; +        BitField<16, 1, u32> InitialLaunchTimestampFlag; +    }; +}; +static_assert(sizeof(InitialLaunchFlag) == 4, "InitialLaunchFlag is an invalid size"); + +/// This is nn::settings::system::SleepFlag +struct SleepFlag { +    union { +        u32 raw{}; + +        BitField<0, 1, u32> SleepsWhilePlayingMedia; +        BitField<1, 1, u32> WakesAtPowerStateChange; +    }; +}; +static_assert(sizeof(SleepFlag) == 4, "TvFlag is an invalid size"); + +/// This is nn::settings::system::NotificationFlag +struct NotificationFlag { +    union { +        u32 raw{}; + +        BitField<0, 1, u32> RingtoneFlag; +        BitField<1, 1, u32> DownloadCompletionFlag; +        BitField<8, 1, u32> EnablesNews; +        BitField<9, 1, u32> IncomingLampFlag; +    }; +}; +static_assert(sizeof(NotificationFlag) == 4, "NotificationFlag is an invalid size"); + +/// This is nn::settings::system::TvFlag +struct TvFlag { +    union { +        u32 raw{}; + +        BitField<0, 1, u32> Allows4k; +        BitField<1, 1, u32> Allows3d; +        BitField<2, 1, u32> AllowsCec; +        BitField<3, 1, u32> PreventsScreenBurnIn; +    }; +}; +static_assert(sizeof(TvFlag) == 4, "TvFlag is an invalid size"); + +/// This is nn::settings::system::UserSelectorFlag +struct UserSelectorFlag { +    union { +        u32 raw{}; + +        BitField<0, 1, u32> SkipIfSingleUser; +        BitField<31, 1, u32> Unknown; +    }; +}; +static_assert(sizeof(UserSelectorFlag) == 4, "UserSelectorFlag is an invalid size"); + +/// This is nn::settings::system::AccountNotificationSettings +struct AccountNotificationSettings { +    Common::UUID uid; +    AccountNotificationFlag flags; +    FriendPresenceOverlayPermission friend_presence_permission; +    FriendPresenceOverlayPermission friend_invitation_permission; +    INSERT_PADDING_BYTES(0x2); +}; +static_assert(sizeof(AccountNotificationSettings) == 0x18, +              "AccountNotificationSettings is an invalid size"); + +/// This is nn::settings::system::EulaVersion +struct EulaVersion { +    u32 version; +    SystemRegionCode region_code; +    EulaVersionClockType clock_type; +    INSERT_PADDING_BYTES(0x4); +    s64 posix_time; +    Time::Clock::SteadyClockTimePoint timestamp; +}; +static_assert(sizeof(EulaVersion) == 0x30, "EulaVersion is incorrect size"); + +struct FirmwareVersionFormat { +    u8 major; +    u8 minor; +    u8 micro; +    INSERT_PADDING_BYTES(1); +    u8 revision_major; +    u8 revision_minor; +    INSERT_PADDING_BYTES(2); +    std::array<char, 0x20> platform; +    std::array<u8, 0x40> version_hash; +    std::array<char, 0x18> display_version; +    std::array<char, 0x80> display_title; +}; +static_assert(sizeof(FirmwareVersionFormat) == 0x100, "FirmwareVersionFormat is an invalid size"); + +/// This is nn::settings::system::HomeMenuScheme +struct HomeMenuScheme { +    u32 main; +    u32 back; +    u32 sub; +    u32 bezel; +    u32 extra; +}; +static_assert(sizeof(HomeMenuScheme) == 0x14, "HomeMenuScheme is incorrect size"); + +/// This is nn::settings::system::InitialLaunchSettings +struct InitialLaunchSettings { +    InitialLaunchFlag flags; +    INSERT_PADDING_BYTES(0x4); +    Service::Time::Clock::SteadyClockTimePoint timestamp; +}; +static_assert(sizeof(InitialLaunchSettings) == 0x20, "InitialLaunchSettings is incorrect size"); + +#pragma pack(push, 4) +struct InitialLaunchSettingsPacked { +    InitialLaunchFlag flags; +    Service::Time::Clock::SteadyClockTimePoint timestamp; +}; +#pragma pack(pop) +static_assert(sizeof(InitialLaunchSettingsPacked) == 0x1C, +              "InitialLaunchSettingsPacked is incorrect size"); + +/// This is nn::settings::system::NotificationTime +struct NotificationTime { +    u32 hour; +    u32 minute; +}; +static_assert(sizeof(NotificationTime) == 0x8, "NotificationTime is an invalid size"); + +/// This is nn::settings::system::NotificationSettings +struct NotificationSettings { +    NotificationFlag flags; +    NotificationVolume volume; +    NotificationTime start_time; +    NotificationTime stop_time; +}; +static_assert(sizeof(NotificationSettings) == 0x18, "NotificationSettings is an invalid size"); + +/// This is nn::settings::system::SleepSettings +struct SleepSettings { +    SleepFlag flags; +    HandheldSleepPlan handheld_sleep_plan; +    ConsoleSleepPlan console_sleep_plan; +}; +static_assert(sizeof(SleepSettings) == 0xc, "SleepSettings is incorrect size"); + +/// This is nn::settings::system::TvSettings +struct TvSettings { +    TvFlag flags; +    TvResolution tv_resolution; +    HdmiContentType hdmi_content_type; +    RgbRange rgb_range; +    CmuMode cmu_mode; +    u32 tv_underscan; +    f32 tv_gama; +    f32 contrast_ratio; +}; +static_assert(sizeof(TvSettings) == 0x20, "TvSettings is an invalid size"); + +} // namespace Service::Set diff --git a/src/core/hle/service/set/system_settings_server.cpp b/src/core/hle/service/set/system_settings_server.cpp index af9348522..87242ae68 100644 --- a/src/core/hle/service/set/system_settings_server.cpp +++ b/src/core/hle/service/set/system_settings_server.cpp @@ -97,8 +97,8 @@ ISystemSettingsServer::ISystemSettingsServer(Core::System& system_)          {3, &ISystemSettingsServer::GetFirmwareVersion, "GetFirmwareVersion"},          {4, &ISystemSettingsServer::GetFirmwareVersion2, "GetFirmwareVersion2"},          {5, nullptr, "GetFirmwareVersionDigest"}, -        {7, nullptr, "GetLockScreenFlag"}, -        {8, nullptr, "SetLockScreenFlag"}, +        {7, &ISystemSettingsServer::GetLockScreenFlag, "GetLockScreenFlag"}, +        {8, &ISystemSettingsServer::SetLockScreenFlag, "SetLockScreenFlag"},          {9, nullptr, "GetBacklightSettings"},          {10, nullptr, "SetBacklightSettings"},          {11, nullptr, "SetBluetoothDevicesSettings"}, @@ -157,12 +157,12 @@ ISystemSettingsServer::ISystemSettingsServer(Core::System& system_)          {66, nullptr, "SetUsb30EnableFlag"},          {67, nullptr, "GetBatteryLot"},          {68, nullptr, "GetSerialNumber"}, -        {69, nullptr, "GetNfcEnableFlag"}, -        {70, nullptr, "SetNfcEnableFlag"}, +        {69, &ISystemSettingsServer::GetNfcEnableFlag, "GetNfcEnableFlag"}, +        {70, &ISystemSettingsServer::SetNfcEnableFlag, "SetNfcEnableFlag"},          {71, &ISystemSettingsServer::GetSleepSettings, "GetSleepSettings"},          {72, &ISystemSettingsServer::SetSleepSettings, "SetSleepSettings"}, -        {73, nullptr, "GetWirelessLanEnableFlag"}, -        {74, nullptr, "SetWirelessLanEnableFlag"}, +        {73, &ISystemSettingsServer::GetWirelessLanEnableFlag, "GetWirelessLanEnableFlag"}, +        {74, &ISystemSettingsServer::SetWirelessLanEnableFlag, "SetWirelessLanEnableFlag"},          {75, &ISystemSettingsServer::GetInitialLaunchSettings, "GetInitialLaunchSettings"},          {76, &ISystemSettingsServer::SetInitialLaunchSettings, "SetInitialLaunchSettings"},          {77, &ISystemSettingsServer::GetDeviceNickName, "GetDeviceNickName"}, @@ -176,8 +176,8 @@ ISystemSettingsServer::ISystemSettingsServer(Core::System& system_)          {85, nullptr, "SetPtmBatteryLot"},          {86, nullptr, "GetPtmFuelGaugeParameter"},          {87, nullptr, "SetPtmFuelGaugeParameter"}, -        {88, nullptr, "GetBluetoothEnableFlag"}, -        {89, nullptr, "SetBluetoothEnableFlag"}, +        {88, &ISystemSettingsServer::GetBluetoothEnableFlag, "GetBluetoothEnableFlag"}, +        {89, &ISystemSettingsServer::SetBluetoothEnableFlag, "SetBluetoothEnableFlag"},          {90, &ISystemSettingsServer::GetMiiAuthorId, "GetMiiAuthorId"},          {91, nullptr, "SetShutdownRtcValue"},          {92, nullptr, "GetShutdownRtcValue"}, @@ -510,6 +510,25 @@ void ISystemSettingsServer::SetUserSystemClockContext(HLERequestContext& ctx) {      rb.Push(res);  } +void ISystemSettingsServer::GetLockScreenFlag(HLERequestContext& ctx) { +    LOG_INFO(Service_SET, "called, lock_screen_flag={}", m_system_settings.lock_screen_flag); + +    IPC::ResponseBuilder rb{ctx, 3}; +    rb.Push(ResultSuccess); +    rb.Push(m_system_settings.lock_screen_flag); +} + +void ISystemSettingsServer::SetLockScreenFlag(HLERequestContext& ctx) { +    IPC::RequestParser rp{ctx}; +    m_system_settings.lock_screen_flag = rp.Pop<bool>(); +    SetSaveNeeded(); + +    LOG_INFO(Service_SET, "called, lock_screen_flag={}", m_system_settings.lock_screen_flag); + +    IPC::ResponseBuilder rb{ctx, 2}; +    rb.Push(ResultSuccess); +} +  void ISystemSettingsServer::GetAccountSettings(HLERequestContext& ctx) {      LOG_INFO(Service_SET, "called"); @@ -531,7 +550,7 @@ void ISystemSettingsServer::SetAccountSettings(HLERequestContext& ctx) {  }  void ISystemSettingsServer::GetEulaVersions(HLERequestContext& ctx) { -    LOG_INFO(Service_SET, "called"); +    LOG_INFO(Service_SET, "called, elements={}", m_system_settings.eula_version_count);      ctx.WriteBuffer(m_system_settings.eula_versions); @@ -557,7 +576,7 @@ void ISystemSettingsServer::SetEulaVersions(HLERequestContext& ctx) {  }  void ISystemSettingsServer::GetColorSetId(HLERequestContext& ctx) { -    LOG_DEBUG(Service_SET, "called"); +    LOG_DEBUG(Service_SET, "called, color_set=", m_system_settings.color_set_id);      IPC::ResponseBuilder rb{ctx, 3};      rb.Push(ResultSuccess); @@ -576,7 +595,13 @@ void ISystemSettingsServer::SetColorSetId(HLERequestContext& ctx) {  }  void ISystemSettingsServer::GetNotificationSettings(HLERequestContext& ctx) { -    LOG_INFO(Service_SET, "called"); +    LOG_INFO(Service_SET, "called, flags={}, volume={}, head_time={}:{}, tailt_time={}:{}", +             m_system_settings.notification_settings.flags.raw, +             m_system_settings.notification_settings.volume, +             m_system_settings.notification_settings.start_time.hour, +             m_system_settings.notification_settings.start_time.minute, +             m_system_settings.notification_settings.stop_time.hour, +             m_system_settings.notification_settings.stop_time.minute);      IPC::ResponseBuilder rb{ctx, 8};      rb.Push(ResultSuccess); @@ -601,7 +626,8 @@ void ISystemSettingsServer::SetNotificationSettings(HLERequestContext& ctx) {  }  void ISystemSettingsServer::GetAccountNotificationSettings(HLERequestContext& ctx) { -    LOG_INFO(Service_SET, "called"); +    LOG_INFO(Service_SET, "called, elements={}", +             m_system_settings.account_notification_settings_count);      ctx.WriteBuffer(m_system_settings.account_notification_settings); @@ -645,6 +671,7 @@ using Settings =  static Settings GetSettings() {      Settings ret; +    // AM      ret["hbloader"]["applet_heap_size"] = ToBytes(u64{0x0});      ret["hbloader"]["applet_heap_reservation_size"] = ToBytes(u64{0x8600000}); @@ -656,6 +683,24 @@ static Settings GetSettings() {      ret["time"]["standard_steady_clock_test_offset_minutes"] = ToBytes(s32{0});      ret["time"]["standard_user_clock_initial_year"] = ToBytes(s32{2023}); +    // HID +    ret["hid_debug"]["enables_debugpad"] = ToBytes(bool{true}); +    ret["hid_debug"]["manages_devices"] = ToBytes(bool{true}); +    ret["hid_debug"]["manages_touch_ic_i2c"] = ToBytes(bool{true}); +    ret["hid_debug"]["emulate_future_device"] = ToBytes(bool{false}); +    ret["hid_debug"]["emulate_mcu_hardware_error"] = ToBytes(bool{false}); +    ret["hid_debug"]["enables_rail"] = ToBytes(bool{true}); +    ret["hid_debug"]["emulate_firmware_update_failure"] = ToBytes(bool{false}); +    ret["hid_debug"]["failure_firmware_update"] = ToBytes(s32{0}); +    ret["hid_debug"]["ble_disabled"] = ToBytes(bool{false}); +    ret["hid_debug"]["dscale_disabled"] = ToBytes(bool{false}); +    ret["hid_debug"]["force_handheld"] = ToBytes(bool{true}); +    ret["hid_debug"]["disabled_features_per_id"] = std::vector<u8>(0xa8); +    ret["hid_debug"]["touch_firmware_auto_update_disabled"] = ToBytes(bool{false}); + +    // Settings +    ret["settings_debug"]["is_debug_mode_enabled"] = ToBytes(bool{false}); +      return ret;  } @@ -708,7 +753,15 @@ void ISystemSettingsServer::GetSettingsItemValue(HLERequestContext& ctx) {  }  void ISystemSettingsServer::GetTvSettings(HLERequestContext& ctx) { -    LOG_INFO(Service_SET, "called"); +    LOG_INFO(Service_SET, +             "called, flags={}, cmu_mode={}, contrast_ratio={}, hdmi_content_type={}, " +             "rgb_range={}, tv_gama={}, tv_resolution={}, tv_underscan={}", +             m_system_settings.tv_settings.flags.raw, m_system_settings.tv_settings.cmu_mode, +             m_system_settings.tv_settings.contrast_ratio, +             m_system_settings.tv_settings.hdmi_content_type, +             m_system_settings.tv_settings.rgb_range, m_system_settings.tv_settings.tv_gama, +             m_system_settings.tv_settings.tv_resolution, +             m_system_settings.tv_settings.tv_underscan);      IPC::ResponseBuilder rb{ctx, 10};      rb.Push(ResultSuccess); @@ -735,23 +788,26 @@ void ISystemSettingsServer::SetTvSettings(HLERequestContext& ctx) {  }  void ISystemSettingsServer::GetDebugModeFlag(HLERequestContext& ctx) { -    LOG_DEBUG(Service_SET, "called"); +    bool is_debug_mode_enabled = false; +    GetSettingsItemValue<bool>(is_debug_mode_enabled, "settings_debug", "is_debug_mode_enabled"); + +    LOG_DEBUG(Service_SET, "called, is_debug_mode_enabled={}", is_debug_mode_enabled);      IPC::ResponseBuilder rb{ctx, 3};      rb.Push(ResultSuccess); -    rb.Push<u32>(0); +    rb.Push(is_debug_mode_enabled);  }  void ISystemSettingsServer::GetQuestFlag(HLERequestContext& ctx) { -    LOG_WARNING(Service_SET, "(STUBBED) called"); +    LOG_INFO(Service_SET, "called, quest_flag={}", m_system_settings.quest_flag);      IPC::ResponseBuilder rb{ctx, 3};      rb.Push(ResultSuccess); -    rb.PushEnum(QuestFlag::Retail); +    rb.PushEnum(m_system_settings.quest_flag);  }  void ISystemSettingsServer::GetDeviceTimeZoneLocationName(HLERequestContext& ctx) { -    LOG_WARNING(Service_SET, "called"); +    LOG_INFO(Service_SET, "called");      Service::Time::TimeZone::LocationName name{};      auto res = GetDeviceTimeZoneLocationName(name); @@ -762,7 +818,7 @@ void ISystemSettingsServer::GetDeviceTimeZoneLocationName(HLERequestContext& ctx  }  void ISystemSettingsServer::SetDeviceTimeZoneLocationName(HLERequestContext& ctx) { -    LOG_WARNING(Service_SET, "called"); +    LOG_INFO(Service_SET, "called");      IPC::RequestParser rp{ctx};      auto name{rp.PopRaw<Service::Time::TimeZone::LocationName>()}; @@ -775,7 +831,7 @@ void ISystemSettingsServer::SetDeviceTimeZoneLocationName(HLERequestContext& ctx  void ISystemSettingsServer::SetRegionCode(HLERequestContext& ctx) {      IPC::RequestParser rp{ctx}; -    m_system_settings.region_code = rp.PopEnum<RegionCode>(); +    m_system_settings.region_code = rp.PopEnum<SystemRegionCode>();      SetSaveNeeded();      LOG_INFO(Service_SET, "called, region_code={}", m_system_settings.region_code); @@ -832,15 +888,38 @@ void ISystemSettingsServer::SetUserSystemClockAutomaticCorrectionEnabled(HLERequ  }  void ISystemSettingsServer::GetPrimaryAlbumStorage(HLERequestContext& ctx) { -    LOG_WARNING(Service_SET, "(STUBBED) called"); +    LOG_INFO(Service_SET, "called, primary_album_storage={}", +             m_system_settings.primary_album_storage); + +    IPC::ResponseBuilder rb{ctx, 3}; +    rb.Push(ResultSuccess); +    rb.PushEnum(m_system_settings.primary_album_storage); +} + +void ISystemSettingsServer::GetNfcEnableFlag(HLERequestContext& ctx) { +    LOG_INFO(Service_SET, "called, nfc_enable_flag={}", m_system_settings.nfc_enable_flag);      IPC::ResponseBuilder rb{ctx, 3};      rb.Push(ResultSuccess); -    rb.PushEnum(PrimaryAlbumStorage::SdCard); +    rb.Push<u8>(m_system_settings.nfc_enable_flag); +} + +void ISystemSettingsServer::SetNfcEnableFlag(HLERequestContext& ctx) { +    IPC::RequestParser rp{ctx}; +    m_system_settings.nfc_enable_flag = rp.Pop<bool>(); +    SetSaveNeeded(); + +    LOG_INFO(Service_SET, "called, nfc_enable_flag={}", m_system_settings.nfc_enable_flag); + +    IPC::ResponseBuilder rb{ctx, 2}; +    rb.Push(ResultSuccess);  }  void ISystemSettingsServer::GetSleepSettings(HLERequestContext& ctx) { -    LOG_INFO(Service_SET, "called"); +    LOG_INFO(Service_SET, "called, flags={}, handheld_sleep_plan={}, console_sleep_plan={}", +             m_system_settings.sleep_settings.flags.raw, +             m_system_settings.sleep_settings.handheld_sleep_plan, +             m_system_settings.sleep_settings.console_sleep_plan);      IPC::ResponseBuilder rb{ctx, 5};      rb.Push(ResultSuccess); @@ -861,8 +940,32 @@ void ISystemSettingsServer::SetSleepSettings(HLERequestContext& ctx) {      rb.Push(ResultSuccess);  } +void ISystemSettingsServer::GetWirelessLanEnableFlag(HLERequestContext& ctx) { +    LOG_INFO(Service_SET, "called, wireless_lan_enable_flag={}", +             m_system_settings.wireless_lan_enable_flag); + +    IPC::ResponseBuilder rb{ctx, 3}; +    rb.Push(ResultSuccess); +    rb.Push(m_system_settings.wireless_lan_enable_flag); +} + +void ISystemSettingsServer::SetWirelessLanEnableFlag(HLERequestContext& ctx) { +    IPC::RequestParser rp{ctx}; +    m_system_settings.wireless_lan_enable_flag = rp.Pop<bool>(); +    SetSaveNeeded(); + +    LOG_INFO(Service_SET, "called, wireless_lan_enable_flag={}", +             m_system_settings.wireless_lan_enable_flag); + +    IPC::ResponseBuilder rb{ctx, 2}; +    rb.Push(ResultSuccess); +} +  void ISystemSettingsServer::GetInitialLaunchSettings(HLERequestContext& ctx) { -    LOG_INFO(Service_SET, "called"); +    LOG_INFO(Service_SET, "called, flags={}, timestamp={}", +             m_system_settings.initial_launch_settings_packed.flags.raw, +             m_system_settings.initial_launch_settings_packed.timestamp.time_point); +      IPC::ResponseBuilder rb{ctx, 10};      rb.Push(ResultSuccess);      rb.PushRaw(m_system_settings.initial_launch_settings_packed); @@ -913,35 +1016,51 @@ void ISystemSettingsServer::GetProductModel(HLERequestContext& ctx) {      rb.Push(product_model);  } -void ISystemSettingsServer::GetMiiAuthorId(HLERequestContext& ctx) { -    const auto author_id = Common::UUID::MakeDefault(); +void ISystemSettingsServer::GetBluetoothEnableFlag(HLERequestContext& ctx) { +    LOG_INFO(Service_SET, "called, bluetooth_enable_flag={}", +             m_system_settings.bluetooth_enable_flag); -    LOG_WARNING(Service_SET, "(STUBBED) called, author_id={}", author_id.FormattedString()); +    IPC::ResponseBuilder rb{ctx, 3}; +    rb.Push(ResultSuccess); +    rb.Push<u8>(m_system_settings.bluetooth_enable_flag); +} + +void ISystemSettingsServer::SetBluetoothEnableFlag(HLERequestContext& ctx) { +    IPC::RequestParser rp{ctx}; +    m_system_settings.bluetooth_enable_flag = rp.Pop<bool>(); +    SetSaveNeeded(); + +    LOG_INFO(Service_SET, "called, bluetooth_enable_flag={}", +             m_system_settings.bluetooth_enable_flag); + +    IPC::ResponseBuilder rb{ctx, 2}; +    rb.Push(ResultSuccess); +} + +void ISystemSettingsServer::GetMiiAuthorId(HLERequestContext& ctx) { +    LOG_INFO(Service_SET, "called, author_id={}", +             m_system_settings.mii_author_id.FormattedString());      IPC::ResponseBuilder rb{ctx, 6};      rb.Push(ResultSuccess); -    rb.PushRaw(author_id); +    rb.PushRaw(m_system_settings.mii_author_id);  }  void ISystemSettingsServer::GetAutoUpdateEnableFlag(HLERequestContext& ctx) { -    u8 auto_update_flag{}; - -    LOG_WARNING(Service_SET, "(STUBBED) called, auto_update_flag={}", auto_update_flag); +    LOG_INFO(Service_SET, "called, auto_update_flag={}", m_system_settings.auto_update_enable_flag);      IPC::ResponseBuilder rb{ctx, 3};      rb.Push(ResultSuccess); -    rb.Push(auto_update_flag); +    rb.Push(m_system_settings.auto_update_enable_flag);  }  void ISystemSettingsServer::GetBatteryPercentageFlag(HLERequestContext& ctx) { -    u8 battery_percentage_flag{1}; - -    LOG_WARNING(Service_SET, "(STUBBED) called, battery_percentage_flag={}", -                battery_percentage_flag); +    LOG_DEBUG(Service_SET, "called, battery_percentage_flag={}", +              m_system_settings.battery_percentage_flag);      IPC::ResponseBuilder rb{ctx, 3};      rb.Push(ResultSuccess); -    rb.Push(battery_percentage_flag); +    rb.Push(m_system_settings.battery_percentage_flag);  }  void ISystemSettingsServer::SetExternalSteadyClockInternalOffset(HLERequestContext& ctx) { @@ -968,11 +1087,12 @@ void ISystemSettingsServer::GetExternalSteadyClockInternalOffset(HLERequestConte  }  void ISystemSettingsServer::GetErrorReportSharePermission(HLERequestContext& ctx) { -    LOG_WARNING(Service_SET, "(STUBBED) called"); +    LOG_INFO(Service_SET, "called, error_report_share_permission={}", +             m_system_settings.error_report_share_permission);      IPC::ResponseBuilder rb{ctx, 3};      rb.Push(ResultSuccess); -    rb.PushEnum(ErrorReportSharePermission::Denied); +    rb.PushEnum(m_system_settings.error_report_share_permission);  }  void ISystemSettingsServer::GetAppletLaunchFlags(HLERequestContext& ctx) { @@ -1014,7 +1134,7 @@ void ISystemSettingsServer::GetKeyboardLayout(HLERequestContext& ctx) {  }  void ISystemSettingsServer::GetDeviceTimeZoneLocationUpdatedTime(HLERequestContext& ctx) { -    LOG_WARNING(Service_SET, "called."); +    LOG_INFO(Service_SET, "called");      Service::Time::Clock::SteadyClockTimePoint time_point{};      auto res = GetDeviceTimeZoneLocationUpdatedTime(time_point); @@ -1025,7 +1145,7 @@ void ISystemSettingsServer::GetDeviceTimeZoneLocationUpdatedTime(HLERequestConte  }  void ISystemSettingsServer::SetDeviceTimeZoneLocationUpdatedTime(HLERequestContext& ctx) { -    LOG_WARNING(Service_SET, "called."); +    LOG_INFO(Service_SET, "called");      IPC::RequestParser rp{ctx};      auto time_point{rp.PopRaw<Service::Time::Clock::SteadyClockTimePoint>()}; @@ -1038,7 +1158,7 @@ void ISystemSettingsServer::SetDeviceTimeZoneLocationUpdatedTime(HLERequestConte  void ISystemSettingsServer::GetUserSystemClockAutomaticCorrectionUpdatedTime(      HLERequestContext& ctx) { -    LOG_WARNING(Service_SET, "called."); +    LOG_INFO(Service_SET, "called");      Service::Time::Clock::SteadyClockTimePoint time_point{};      auto res = GetUserSystemClockAutomaticCorrectionUpdatedTime(time_point); @@ -1050,7 +1170,7 @@ void ISystemSettingsServer::GetUserSystemClockAutomaticCorrectionUpdatedTime(  void ISystemSettingsServer::SetUserSystemClockAutomaticCorrectionUpdatedTime(      HLERequestContext& ctx) { -    LOG_WARNING(Service_SET, "called."); +    LOG_INFO(Service_SET, "called");      IPC::RequestParser rp{ctx};      auto time_point{rp.PopRaw<Service::Time::Clock::SteadyClockTimePoint>()}; @@ -1062,11 +1182,12 @@ void ISystemSettingsServer::SetUserSystemClockAutomaticCorrectionUpdatedTime(  }  void ISystemSettingsServer::GetChineseTraditionalInputMethod(HLERequestContext& ctx) { -    LOG_WARNING(Service_SET, "(STUBBED) called"); +    LOG_INFO(Service_SET, "called, chinese_traditional_input_method={}", +             m_system_settings.chinese_traditional_input_method);      IPC::ResponseBuilder rb{ctx, 3};      rb.Push(ResultSuccess); -    rb.PushEnum(ChineseTraditionalInputMethod::Unknown0); +    rb.PushEnum(m_system_settings.chinese_traditional_input_method);  }  void ISystemSettingsServer::GetHomeMenuScheme(HLERequestContext& ctx) { @@ -1094,11 +1215,11 @@ void ISystemSettingsServer::GetHomeMenuSchemeModel(HLERequestContext& ctx) {  }  void ISystemSettingsServer::GetFieldTestingFlag(HLERequestContext& ctx) { -    LOG_WARNING(Service_SET, "(STUBBED) called"); +    LOG_INFO(Service_SET, "called, field_testing_flag={}", m_system_settings.field_testing_flag);      IPC::ResponseBuilder rb{ctx, 3};      rb.Push(ResultSuccess); -    rb.Push<u8>(false); +    rb.Push(m_system_settings.field_testing_flag);  }  void ISystemSettingsServer::SetupSettings() { diff --git a/src/core/hle/service/set/system_settings_server.h b/src/core/hle/service/set/system_settings_server.h index 6f587e0b3..32716f567 100644 --- a/src/core/hle/service/set/system_settings_server.h +++ b/src/core/hle/service/set/system_settings_server.h @@ -12,10 +12,11 @@  #include "common/uuid.h"  #include "core/hle/result.h"  #include "core/hle/service/service.h" -#include "core/hle/service/set/appln_settings.h" -#include "core/hle/service/set/device_settings.h" -#include "core/hle/service/set/private_settings.h" -#include "core/hle/service/set/system_settings.h" +#include "core/hle/service/set/setting_formats/appln_settings.h" +#include "core/hle/service/set/setting_formats/device_settings.h" +#include "core/hle/service/set/setting_formats/private_settings.h" +#include "core/hle/service/set/setting_formats/system_settings.h" +#include "core/hle/service/set/settings_types.h"  #include "core/hle/service/time/clock_types.h"  #include "core/hle/service/time/time_zone_types.h" @@ -24,25 +25,6 @@ class System;  }  namespace Service::Set { -enum class GetFirmwareVersionType { -    Version1, -    Version2, -}; - -struct FirmwareVersionFormat { -    u8 major; -    u8 minor; -    u8 micro; -    INSERT_PADDING_BYTES(1); -    u8 revision_major; -    u8 revision_minor; -    INSERT_PADDING_BYTES(2); -    std::array<char, 0x20> platform; -    std::array<u8, 0x40> version_hash; -    std::array<char, 0x18> display_version; -    std::array<char, 0x80> display_title; -}; -static_assert(sizeof(FirmwareVersionFormat) == 0x100, "FirmwareVersionFormat is an invalid size");  Result GetFirmwareVersionImpl(FirmwareVersionFormat& out_firmware, Core::System& system,                                GetFirmwareVersionType type); @@ -55,6 +37,18 @@ public:      Result GetSettingsItemValue(std::vector<u8>& out_value, const std::string& category,                                  const std::string& name); +    template <typename T> +    Result GetSettingsItemValue(T& value, const std::string& category, const std::string& name) { +        std::vector<u8> data; +        const auto result = GetSettingsItemValue(data, category, name); +        if (result.IsError()) { +            return result; +        } +        ASSERT(data.size() >= sizeof(T)); +        std::memcpy(&value, data.data(), sizeof(T)); +        return result; +    } +      Result GetExternalSteadyClockSourceId(Common::UUID& out_id);      Result SetExternalSteadyClockSourceId(Common::UUID id);      Result GetUserSystemClockContext(Service::Time::Clock::SystemClockContext& out_context); @@ -80,6 +74,8 @@ private:      void SetLanguageCode(HLERequestContext& ctx);      void GetFirmwareVersion(HLERequestContext& ctx);      void GetFirmwareVersion2(HLERequestContext& ctx); +    void GetLockScreenFlag(HLERequestContext& ctx); +    void SetLockScreenFlag(HLERequestContext& ctx);      void GetExternalSteadyClockSourceId(HLERequestContext& ctx);      void SetExternalSteadyClockSourceId(HLERequestContext& ctx);      void GetUserSystemClockContext(HLERequestContext& ctx); @@ -108,13 +104,19 @@ private:      void IsUserSystemClockAutomaticCorrectionEnabled(HLERequestContext& ctx);      void SetUserSystemClockAutomaticCorrectionEnabled(HLERequestContext& ctx);      void GetPrimaryAlbumStorage(HLERequestContext& ctx); +    void GetNfcEnableFlag(HLERequestContext& ctx); +    void SetNfcEnableFlag(HLERequestContext& ctx);      void GetSleepSettings(HLERequestContext& ctx);      void SetSleepSettings(HLERequestContext& ctx); +    void GetWirelessLanEnableFlag(HLERequestContext& ctx); +    void SetWirelessLanEnableFlag(HLERequestContext& ctx);      void GetInitialLaunchSettings(HLERequestContext& ctx);      void SetInitialLaunchSettings(HLERequestContext& ctx);      void GetDeviceNickName(HLERequestContext& ctx);      void SetDeviceNickName(HLERequestContext& ctx);      void GetProductModel(HLERequestContext& ctx); +    void GetBluetoothEnableFlag(HLERequestContext& ctx); +    void SetBluetoothEnableFlag(HLERequestContext& ctx);      void GetMiiAuthorId(HLERequestContext& ctx);      void GetAutoUpdateEnableFlag(HLERequestContext& ctx);      void GetBatteryPercentageFlag(HLERequestContext& ctx); diff --git a/src/hid_core/frontend/emulated_controller.cpp b/src/hid_core/frontend/emulated_controller.cpp index 2ab93402d..b001cf2fd 100644 --- a/src/hid_core/frontend/emulated_controller.cpp +++ b/src/hid_core/frontend/emulated_controller.cpp @@ -110,6 +110,7 @@ void EmulatedController::ReloadFromSettings() {          original_npad_type = npad_type;      } +    SetPollingMode(EmulatedDeviceIndex::RightIndex, Common::Input::PollingMode::Active);      Disconnect();      if (player.connected) {          Connect(); diff --git a/src/hid_core/resources/hid_firmware_settings.cpp b/src/hid_core/resources/hid_firmware_settings.cpp index 9fa0db17e..00ceff7e6 100644 --- a/src/hid_core/resources/hid_firmware_settings.cpp +++ b/src/hid_core/resources/hid_firmware_settings.cpp @@ -1,11 +1,14 @@  // SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project  // SPDX-License-Identifier: GPL-3.0-or-later +#include "core/hle/service/set/system_settings_server.h" +#include "core/hle/service/sm/sm.h"  #include "hid_core/resources/hid_firmware_settings.h"  namespace Service::HID { -HidFirmwareSettings::HidFirmwareSettings() { +HidFirmwareSettings::HidFirmwareSettings(Core::System& system) { +    m_set_sys = system.ServiceManager().GetService<Service::Set::ISystemSettingsServer>("set:sys");      LoadSettings(true);  } @@ -18,21 +21,25 @@ void HidFirmwareSettings::LoadSettings(bool reload_config) {          return;      } -    // TODO: Use nn::settings::fwdbg::GetSettingsItemValue to load config values - -    is_debug_pad_enabled = true; -    is_device_managed = true; -    is_touch_i2c_managed = is_device_managed; -    is_future_devices_emulated = false; -    is_mcu_hardware_error_emulated = false; -    is_rail_enabled = true; -    is_firmware_update_failure_emulated = false; +    m_set_sys->GetSettingsItemValue<bool>(is_debug_pad_enabled, "hid_debug", "enables_debugpad"); +    m_set_sys->GetSettingsItemValue<bool>(is_device_managed, "hid_debug", "manages_devices"); +    m_set_sys->GetSettingsItemValue<bool>(is_touch_i2c_managed, "hid_debug", +                                          "manages_touch_ic_i2c"); +    m_set_sys->GetSettingsItemValue<bool>(is_future_devices_emulated, "hid_debug", +                                          "emulate_future_device"); +    m_set_sys->GetSettingsItemValue<bool>(is_mcu_hardware_error_emulated, "hid_debug", +                                          "emulate_mcu_hardware_error"); +    m_set_sys->GetSettingsItemValue<bool>(is_rail_enabled, "hid_debug", "enables_rail"); +    m_set_sys->GetSettingsItemValue<bool>(is_firmware_update_failure_emulated, "hid_debug", +                                          "emulate_firmware_update_failure");      is_firmware_update_failure = {}; -    is_ble_disabled = false; -    is_dscale_disabled = false; -    is_handheld_forced = true; +    m_set_sys->GetSettingsItemValue<bool>(is_ble_disabled, "hid_debug", "ble_disabled"); +    m_set_sys->GetSettingsItemValue<bool>(is_dscale_disabled, "hid_debug", "dscale_disabled"); +    m_set_sys->GetSettingsItemValue<bool>(is_handheld_forced, "hid_debug", "force_handheld");      features_per_id_disabled = {}; -    is_touch_firmware_auto_update_disabled = false; +    m_set_sys->GetSettingsItemValue<bool>(is_touch_firmware_auto_update_disabled, "hid_debug", +                                          "touch_firmware_auto_update_disabled"); +      is_initialized = true;  } diff --git a/src/hid_core/resources/hid_firmware_settings.h b/src/hid_core/resources/hid_firmware_settings.h index 00201fd94..3694fa9a3 100644 --- a/src/hid_core/resources/hid_firmware_settings.h +++ b/src/hid_core/resources/hid_firmware_settings.h @@ -5,6 +5,14 @@  #include "common/common_types.h" +namespace Core { +class System; +} + +namespace Service::Set { +class ISystemSettingsServer; +} +  namespace Service::HID {  /// Loads firmware config from nn::settings::fwdbg @@ -13,7 +21,7 @@ public:      using FirmwareSetting = std::array<u8, 4>;      using FeaturesPerId = std::array<bool, 0xA8>; -    HidFirmwareSettings(); +    HidFirmwareSettings(Core::System& system);      void Reload();      void LoadSettings(bool reload_config); @@ -49,6 +57,8 @@ private:      bool is_touch_firmware_auto_update_disabled{};      FirmwareSetting is_firmware_update_failure{};      FeaturesPerId features_per_id_disabled{}; + +    std::shared_ptr<Service::Set::ISystemSettingsServer> m_set_sys;  };  } // namespace Service::HID diff --git a/src/hid_core/resources/npad/npad.cpp b/src/hid_core/resources/npad/npad.cpp index 97537a2e2..de0f9cbb9 100644 --- a/src/hid_core/resources/npad/npad.cpp +++ b/src/hid_core/resources/npad/npad.cpp @@ -1232,6 +1232,13 @@ Result NPad::RegisterAppletResourceUserId(u64 aruid) {  }  void NPad::UnregisterAppletResourceUserId(u64 aruid) { +    // TODO: Remove this once abstract pad is emulated properly +    const auto aruid_index = npad_resource.GetIndexFromAruid(aruid); +    for (auto& controller : controller_data[aruid_index]) { +        controller.is_connected = false; +        controller.shared_memory = nullptr; +    } +      npad_resource.UnregisterAppletResourceUserId(aruid);  } diff --git a/src/input_common/CMakeLists.txt b/src/input_common/CMakeLists.txt index d2fbea488..d0a71a15b 100644 --- a/src/input_common/CMakeLists.txt +++ b/src/input_common/CMakeLists.txt @@ -2,6 +2,8 @@  # SPDX-License-Identifier: GPL-2.0-or-later  add_library(input_common STATIC +    drivers/android.cpp +    drivers/android.h      drivers/camera.cpp      drivers/camera.h      drivers/keyboard.cpp diff --git a/src/input_common/drivers/android.cpp b/src/input_common/drivers/android.cpp new file mode 100644 index 000000000..b6a03fdc0 --- /dev/null +++ b/src/input_common/drivers/android.cpp @@ -0,0 +1,48 @@ +// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project +// SPDX-License-Identifier: GPL-3.0-or-later + +#include "input_common/drivers/android.h" + +namespace InputCommon { + +Android::Android(std::string input_engine_) : InputEngine(std::move(input_engine_)) {} + +void Android::RegisterController(std::size_t controller_number) { +    PreSetController(GetIdentifier(controller_number)); +} + +void Android::SetButtonState(std::size_t controller_number, int button_id, bool value) { +    const auto identifier = GetIdentifier(controller_number); +    SetButton(identifier, button_id, value); +} + +void Android::SetAxisState(std::size_t controller_number, int axis_id, float value) { +    const auto identifier = GetIdentifier(controller_number); +    SetAxis(identifier, axis_id, value); +} + +void Android::SetMotionState(std::size_t controller_number, u64 delta_timestamp, float gyro_x, +                             float gyro_y, float gyro_z, float accel_x, float accel_y, +                             float accel_z) { +    const auto identifier = GetIdentifier(controller_number); +    const BasicMotion motion_data{ +        .gyro_x = gyro_x, +        .gyro_y = gyro_y, +        .gyro_z = gyro_z, +        .accel_x = accel_x, +        .accel_y = accel_y, +        .accel_z = accel_z, +        .delta_timestamp = delta_timestamp, +    }; +    SetMotion(identifier, 0, motion_data); +} + +PadIdentifier Android::GetIdentifier(std::size_t controller_number) const { +    return { +        .guid = Common::UUID{}, +        .port = controller_number, +        .pad = 0, +    }; +} + +} // namespace InputCommon diff --git a/src/input_common/drivers/android.h b/src/input_common/drivers/android.h new file mode 100644 index 000000000..3f01817f6 --- /dev/null +++ b/src/input_common/drivers/android.h @@ -0,0 +1,54 @@ +// SPDX-FileCopyrightText: Copyright 2024 yuzu Emulator Project +// SPDX-License-Identifier: GPL-3.0-or-later + +#pragma once + +#include "input_common/input_engine.h" + +namespace InputCommon { + +/** + * A virtual controller that is always assigned to the game input + */ +class Android final : public InputEngine { +public: +    explicit Android(std::string input_engine_); + +    /** +     * Registers controller number to accept new inputs +     * @param controller_number the controller number that will take this action +     */ +    void RegisterController(std::size_t controller_number); + +    /** +     * Sets the status of all buttons bound with the key to pressed +     * @param controller_number the controller number that will take this action +     * @param button_id the id of the button +     * @param value indicates if the button is pressed or not +     */ +    void SetButtonState(std::size_t controller_number, int button_id, bool value); + +    /** +     * Sets the status of a analog input to a specific player index +     * @param controller_number the controller number that will take this action +     * @param axis_id the id of the axis to move +     * @param value the analog position of the axis +     */ +    void SetAxisState(std::size_t controller_number, int axis_id, float value); + +    /** +     * Sets the status of the motion sensor to a specific player index +     * @param controller_number the controller number that will take this action +     * @param delta_timestamp time passed since last reading +     * @param gyro_x,gyro_y,gyro_z the gyro sensor readings +     * @param accel_x,accel_y,accel_z the accelerometer reading +     */ +    void SetMotionState(std::size_t controller_number, u64 delta_timestamp, float gyro_x, +                        float gyro_y, float gyro_z, float accel_x, float accel_y, float accel_z); + +private: +    /// Returns the correct identifier corresponding to the player index +    PadIdentifier GetIdentifier(std::size_t controller_number) const; +}; + +} // namespace InputCommon diff --git a/src/input_common/input_mapping.cpp b/src/input_common/input_mapping.cpp index 8c2ee4eb3..f1a1d7398 100644 --- a/src/input_common/input_mapping.cpp +++ b/src/input_common/input_mapping.cpp @@ -210,6 +210,9 @@ bool MappingFactory::IsDriverValid(const MappingData& data) const {      if (data.engine == "analog_from_button") {          return false;      } +    if (data.engine == "virtual_gamepad") { +        return false; +    }      return true;  } diff --git a/src/input_common/main.cpp b/src/input_common/main.cpp index c77fc04ee..f8749ebbf 100644 --- a/src/input_common/main.cpp +++ b/src/input_common/main.cpp @@ -4,6 +4,7 @@  #include <memory>  #include "common/input.h"  #include "common/param_package.h" +#include "input_common/drivers/android.h"  #include "input_common/drivers/camera.h"  #include "input_common/drivers/keyboard.h"  #include "input_common/drivers/mouse.h" @@ -78,6 +79,7 @@ struct InputSubsystem::Impl {          RegisterEngine("cemuhookudp", udp_client);          RegisterEngine("tas", tas_input);          RegisterEngine("camera", camera); +        RegisterEngine("android", android);          RegisterEngine("virtual_amiibo", virtual_amiibo);          RegisterEngine("virtual_gamepad", virtual_gamepad);  #ifdef HAVE_SDL2 @@ -109,6 +111,7 @@ struct InputSubsystem::Impl {          UnregisterEngine(udp_client);          UnregisterEngine(tas_input);          UnregisterEngine(camera); +        UnregisterEngine(android);          UnregisterEngine(virtual_amiibo);          UnregisterEngine(virtual_gamepad);  #ifdef HAVE_SDL2 @@ -129,6 +132,8 @@ struct InputSubsystem::Impl {          devices.insert(devices.end(), keyboard_devices.begin(), keyboard_devices.end());          auto mouse_devices = mouse->GetInputDevices();          devices.insert(devices.end(), mouse_devices.begin(), mouse_devices.end()); +        auto android_devices = android->GetInputDevices(); +        devices.insert(devices.end(), android_devices.begin(), android_devices.end());  #ifdef HAVE_LIBUSB          auto gcadapter_devices = gcadapter->GetInputDevices();          devices.insert(devices.end(), gcadapter_devices.begin(), gcadapter_devices.end()); @@ -157,6 +162,9 @@ struct InputSubsystem::Impl {          if (engine == mouse->GetEngineName()) {              return mouse;          } +        if (engine == android->GetEngineName()) { +            return android; +        }  #ifdef HAVE_LIBUSB          if (engine == gcadapter->GetEngineName()) {              return gcadapter; @@ -237,6 +245,9 @@ struct InputSubsystem::Impl {          if (engine == mouse->GetEngineName()) {              return true;          } +        if (engine == android->GetEngineName()) { +            return true; +        }  #ifdef HAVE_LIBUSB          if (engine == gcadapter->GetEngineName()) {              return true; @@ -265,6 +276,7 @@ struct InputSubsystem::Impl {      void BeginConfiguration() {          keyboard->BeginConfiguration();          mouse->BeginConfiguration(); +        android->BeginConfiguration();  #ifdef HAVE_LIBUSB          gcadapter->BeginConfiguration();  #endif @@ -278,6 +290,7 @@ struct InputSubsystem::Impl {      void EndConfiguration() {          keyboard->EndConfiguration();          mouse->EndConfiguration(); +        android->EndConfiguration();  #ifdef HAVE_LIBUSB          gcadapter->EndConfiguration();  #endif @@ -308,6 +321,7 @@ struct InputSubsystem::Impl {      std::shared_ptr<TasInput::Tas> tas_input;      std::shared_ptr<CemuhookUDP::UDPClient> udp_client;      std::shared_ptr<Camera> camera; +    std::shared_ptr<Android> android;      std::shared_ptr<VirtualAmiibo> virtual_amiibo;      std::shared_ptr<VirtualGamepad> virtual_gamepad; @@ -373,6 +387,14 @@ const Camera* InputSubsystem::GetCamera() const {      return impl->camera.get();  } +Android* InputSubsystem::GetAndroid() { +    return impl->android.get(); +} + +const Android* InputSubsystem::GetAndroid() const { +    return impl->android.get(); +} +  VirtualAmiibo* InputSubsystem::GetVirtualAmiibo() {      return impl->virtual_amiibo.get();  } diff --git a/src/input_common/main.h b/src/input_common/main.h index d64a6cb4c..1d19019ee 100644 --- a/src/input_common/main.h +++ b/src/input_common/main.h @@ -29,6 +29,7 @@ enum Values : int;  }  namespace InputCommon { +class Android;  class Camera;  class Keyboard;  class Mouse; @@ -103,6 +104,12 @@ public:      /// Retrieves the underlying camera input device.      [[nodiscard]] const Camera* GetCamera() const; +    /// Retrieves the underlying android input device. +    [[nodiscard]] Android* GetAndroid(); + +    /// Retrieves the underlying android input device. +    [[nodiscard]] const Android* GetAndroid() const; +      /// Retrieves the underlying virtual amiibo input device.      [[nodiscard]] VirtualAmiibo* GetVirtualAmiibo(); diff --git a/src/shader_recompiler/backend/spirv/emit_spirv_image.cpp b/src/shader_recompiler/backend/spirv/emit_spirv_image.cpp index 800754554..64a4e0e55 100644 --- a/src/shader_recompiler/backend/spirv/emit_spirv_image.cpp +++ b/src/shader_recompiler/backend/spirv/emit_spirv_image.cpp @@ -12,6 +12,11 @@ namespace Shader::Backend::SPIRV {  namespace {  class ImageOperands {  public: +    [[maybe_unused]] static constexpr bool ImageSampleOffsetAllowed = false; +    [[maybe_unused]] static constexpr bool ImageGatherOffsetAllowed = true; +    [[maybe_unused]] static constexpr bool ImageFetchOffsetAllowed = false; +    [[maybe_unused]] static constexpr bool ImageGradientOffsetAllowed = false; +      explicit ImageOperands(EmitContext& ctx, bool has_bias, bool has_lod, bool has_lod_clamp,                             Id lod, const IR::Value& offset) {          if (has_bias) { @@ -22,7 +27,7 @@ public:              const Id lod_value{has_lod_clamp ? ctx.OpCompositeExtract(ctx.F32[1], lod, 0) : lod};              Add(spv::ImageOperandsMask::Lod, lod_value);          } -        AddOffset(ctx, offset); +        AddOffset(ctx, offset, ImageSampleOffsetAllowed);          if (has_lod_clamp) {              const Id lod_clamp{has_bias ? ctx.OpCompositeExtract(ctx.F32[1], lod, 1) : lod};              Add(spv::ImageOperandsMask::MinLod, lod_clamp); @@ -55,20 +60,17 @@ public:          Add(spv::ImageOperandsMask::ConstOffsets, offsets);      } -    explicit ImageOperands(Id offset, Id lod, Id ms) { +    explicit ImageOperands(Id lod, Id ms) {          if (Sirit::ValidId(lod)) {              Add(spv::ImageOperandsMask::Lod, lod);          } -        if (Sirit::ValidId(offset)) { -            Add(spv::ImageOperandsMask::Offset, offset); -        }          if (Sirit::ValidId(ms)) {              Add(spv::ImageOperandsMask::Sample, ms);          }      }      explicit ImageOperands(EmitContext& ctx, bool has_lod_clamp, Id derivatives, -                           u32 num_derivatives, Id offset, Id lod_clamp) { +                           u32 num_derivatives, const IR::Value& offset, Id lod_clamp) {          if (!Sirit::ValidId(derivatives)) {              throw LogicError("Derivatives must be present");          } @@ -83,16 +85,14 @@ public:          const Id derivatives_Y{ctx.OpCompositeConstruct(              ctx.F32[num_derivatives], std::span{deriv_y_accum.data(), deriv_y_accum.size()})};          Add(spv::ImageOperandsMask::Grad, derivatives_X, derivatives_Y); -        if (Sirit::ValidId(offset)) { -            Add(spv::ImageOperandsMask::Offset, offset); -        } +        AddOffset(ctx, offset, ImageGradientOffsetAllowed);          if (has_lod_clamp) {              Add(spv::ImageOperandsMask::MinLod, lod_clamp);          }      }      explicit ImageOperands(EmitContext& ctx, bool has_lod_clamp, Id derivatives_1, Id derivatives_2, -                           Id offset, Id lod_clamp) { +                           const IR::Value& offset, Id lod_clamp) {          if (!Sirit::ValidId(derivatives_1) || !Sirit::ValidId(derivatives_2)) {              throw LogicError("Derivatives must be present");          } @@ -111,9 +111,7 @@ public:          const Id derivatives_id2{ctx.OpCompositeConstruct(              ctx.F32[3], std::span{deriv_2_accum.data(), deriv_2_accum.size()})};          Add(spv::ImageOperandsMask::Grad, derivatives_id1, derivatives_id2); -        if (Sirit::ValidId(offset)) { -            Add(spv::ImageOperandsMask::Offset, offset); -        } +        AddOffset(ctx, offset, ImageGradientOffsetAllowed);          if (has_lod_clamp) {              Add(spv::ImageOperandsMask::MinLod, lod_clamp);          } @@ -132,7 +130,7 @@ public:      }  private: -    void AddOffset(EmitContext& ctx, const IR::Value& offset) { +    void AddOffset(EmitContext& ctx, const IR::Value& offset, bool runtime_offset_allowed) {          if (offset.IsEmpty()) {              return;          } @@ -165,7 +163,9 @@ private:                  break;              }          } -        Add(spv::ImageOperandsMask::Offset, ctx.Def(offset)); +        if (runtime_offset_allowed) { +            Add(spv::ImageOperandsMask::Offset, ctx.Def(offset)); +        }      }      void Add(spv::ImageOperandsMask new_mask, Id value) { @@ -311,6 +311,37 @@ Id ImageGatherSubpixelOffset(EmitContext& ctx, const IR::TextureInstInfo& info,          return coords;      }  } + +void AddOffsetToCoordinates(EmitContext& ctx, const IR::TextureInstInfo& info, Id& coords, +                            Id offset) { +    if (!Sirit::ValidId(offset)) { +        return; +    } + +    Id result_type{}; +    switch (info.type) { +    case TextureType::Buffer: +    case TextureType::Color1D: +    case TextureType::ColorArray1D: { +        result_type = ctx.U32[1]; +        break; +    } +    case TextureType::Color2D: +    case TextureType::Color2DRect: +    case TextureType::ColorArray2D: { +        result_type = ctx.U32[2]; +        break; +    } +    case TextureType::Color3D: { +        result_type = ctx.U32[3]; +        break; +    } +    case TextureType::ColorCube: +    case TextureType::ColorArrayCube: +        return; +    } +    coords = ctx.OpIAdd(result_type, coords, offset); +}  } // Anonymous namespace  Id EmitBindlessImageSampleImplicitLod(EmitContext&) { @@ -496,6 +527,7 @@ Id EmitImageGatherDref(EmitContext& ctx, IR::Inst* inst, const IR::Value& index,  Id EmitImageFetch(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords, Id offset,                    Id lod, Id ms) {      const auto info{inst->Flags<IR::TextureInstInfo>()}; +    AddOffsetToCoordinates(ctx, info, coords, offset);      if (info.type == TextureType::Buffer) {          lod = Id{};      } @@ -503,7 +535,7 @@ Id EmitImageFetch(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id c          // This image is multisampled, lod must be implicit          lod = Id{};      } -    const ImageOperands operands(offset, lod, ms); +    const ImageOperands operands(lod, ms);      return Emit(&EmitContext::OpImageSparseFetch, &EmitContext::OpImageFetch, ctx, inst, ctx.F32[4],                  TextureImage(ctx, info, index), coords, operands.MaskOptional(), operands.Span());  } @@ -548,13 +580,13 @@ Id EmitImageQueryLod(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, I  }  Id EmitImageGradient(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords, -                     Id derivatives, Id offset, Id lod_clamp) { +                     Id derivatives, const IR::Value& offset, Id lod_clamp) {      const auto info{inst->Flags<IR::TextureInstInfo>()}; -    const auto operands = -        info.num_derivatives == 3 -            ? ImageOperands(ctx, info.has_lod_clamp != 0, derivatives, offset, {}, lod_clamp) -            : ImageOperands(ctx, info.has_lod_clamp != 0, derivatives, info.num_derivatives, offset, -                            lod_clamp); +    const auto operands = info.num_derivatives == 3 +                              ? ImageOperands(ctx, info.has_lod_clamp != 0, derivatives, +                                              ctx.Def(offset), {}, lod_clamp) +                              : ImageOperands(ctx, info.has_lod_clamp != 0, derivatives, +                                              info.num_derivatives, offset, lod_clamp);      return Emit(&EmitContext::OpImageSparseSampleExplicitLod,                  &EmitContext::OpImageSampleExplicitLod, ctx, inst, ctx.F32[4],                  Texture(ctx, info, index), coords, operands.Mask(), operands.Span()); diff --git a/src/shader_recompiler/backend/spirv/emit_spirv_instructions.h b/src/shader_recompiler/backend/spirv/emit_spirv_instructions.h index 7d34575c8..5c01b1012 100644 --- a/src/shader_recompiler/backend/spirv/emit_spirv_instructions.h +++ b/src/shader_recompiler/backend/spirv/emit_spirv_instructions.h @@ -543,7 +543,7 @@ Id EmitImageQueryDimensions(EmitContext& ctx, IR::Inst* inst, const IR::Value& i                              const IR::Value& skip_mips);  Id EmitImageQueryLod(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords);  Id EmitImageGradient(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords, -                     Id derivatives, Id offset, Id lod_clamp); +                     Id derivatives, const IR::Value& offset, Id lod_clamp);  Id EmitImageRead(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords);  void EmitImageWrite(EmitContext& ctx, IR::Inst* inst, const IR::Value& index, Id coords, Id color);  Id EmitIsTextureScaled(EmitContext& ctx, const IR::Value& index); diff --git a/src/yuzu/CMakeLists.txt b/src/yuzu/CMakeLists.txt index 90278052a..93b03b917 100644 --- a/src/yuzu/CMakeLists.txt +++ b/src/yuzu/CMakeLists.txt @@ -351,7 +351,7 @@ if (APPLE)      if (NOT USE_SYSTEM_MOLTENVK)          set(MOLTENVK_PLATFORM "macOS") -        set(MOLTENVK_VERSION "v1.2.5") +        set(MOLTENVK_VERSION "v1.2.7")          download_moltenvk_external(${MOLTENVK_PLATFORM} ${MOLTENVK_VERSION})      endif()      find_library(MOLTENVK_LIBRARY MoltenVK REQUIRED) diff --git a/src/yuzu/main.cpp b/src/yuzu/main.cpp index 33756febf..3c562e3b2 100644 --- a/src/yuzu/main.cpp +++ b/src/yuzu/main.cpp @@ -518,12 +518,21 @@ GMainWindow::GMainWindow(std::unique_ptr<QtConfig> config_, bool has_broken_vulk                  continue;              } +            int user_arg_idx = ++i;              bool argument_ok; -            const std::size_t selected_user = args[++i].toUInt(&argument_ok); +            std::size_t selected_user = args[user_arg_idx].toUInt(&argument_ok);              if (!argument_ok) { -                LOG_ERROR(Frontend, "Invalid user argument"); -                continue; +                // try to look it up by username, only finds the first username that matches. +                const std::string user_arg_str = args[user_arg_idx].toStdString(); +                const auto user_idx = system->GetProfileManager().GetUserIndex(user_arg_str); + +                if (user_idx == std::nullopt) { +                    LOG_ERROR(Frontend, "Invalid user argument"); +                    continue; +                } + +                selected_user = user_idx.value();              }              if (!system->GetProfileManager().UserExistsIndex(selected_user)) { @@ -532,6 +541,8 @@ GMainWindow::GMainWindow(std::unique_ptr<QtConfig> config_, bool has_broken_vulk              }              Settings::values.current_user = static_cast<s32>(selected_user); + +            user_flag_cmd_line = true;              continue;          } @@ -1942,7 +1953,7 @@ void GMainWindow::BootGame(const QString& filename, u64 program_id, std::size_t      Settings::LogSettings(); -    if (UISettings::values.select_user_on_boot) { +    if (UISettings::values.select_user_on_boot && !user_flag_cmd_line) {          const Core::Frontend::ProfileSelectParameters parameters{              .mode = Service::AM::Applets::UiMode::UserSelector,              .invalid_uid_list = {}, @@ -1954,6 +1965,11 @@ void GMainWindow::BootGame(const QString& filename, u64 program_id, std::size_t          }      } +    // If the user specifies -u (successfully) on the cmd line, don't prompt for a user on first +    // game startup only. If the user stops emulation and starts a new one, go back to the expected +    // behavior of asking. +    user_flag_cmd_line = false; +      if (!LoadROM(filename, program_id, program_index, launch_type)) {          return;      } diff --git a/src/yuzu/main.h b/src/yuzu/main.h index 366e806d5..f3276da64 100644 --- a/src/yuzu/main.h +++ b/src/yuzu/main.h @@ -523,6 +523,8 @@ private:      std::unique_ptr<EmuThread> emu_thread;      // The path to the game currently running      QString current_game_path; +    // Whether a user was set on the command line (skips UserSelector if it's forced to show up) +    bool user_flag_cmd_line = false;      bool auto_paused = false;      bool auto_muted = false; | 
