From 19d5695aa33a7584737fdcb1c37a149b1d4e7033 Mon Sep 17 00:00:00 2001 From: PabloMK7 Date: Fri, 12 Jan 2024 18:15:47 +0100 Subject: [PATCH] Implement some missing/wrong AC functionality. (#7171) * Implement some missing/wrong AC functionality. * Schedule NDM connect event into the future * Disable NDM connect for now as it's causing issues * Apply latest changes and suggestions. * Workaround to fake wifi connection. * Add missing command to ac:i * Fix compilation * Fix error codes for CamcelConnectAsync * Fix missing global state. --- src/core/hle/kernel/shared_page.cpp | 20 +++ src/core/hle/kernel/shared_page.h | 26 ++- src/core/hle/service/ac/ac.cpp | 253 ++++++++++++++++++++++----- src/core/hle/service/ac/ac.h | 107 ++++++++++- src/core/hle/service/ac/ac_i.cpp | 9 +- src/core/hle/service/ac/ac_u.cpp | 9 +- src/core/hle/service/nwm/nwm_uds.cpp | 9 +- 7 files changed, 375 insertions(+), 58 deletions(-) diff --git a/src/core/hle/kernel/shared_page.cpp b/src/core/hle/kernel/shared_page.cpp index 3583974862..10264c4e95 100644 --- a/src/core/hle/kernel/shared_page.cpp +++ b/src/core/hle/kernel/shared_page.cpp @@ -85,6 +85,12 @@ Handler::Handler(Core::Timing& timing, u64 override_init_time) : timing(timing) float slidestate = Settings::values.factor_3d.GetValue() / 100.0f; shared_page.sliderstate_3d = static_cast(slidestate); + + // TODO(PabloMK7) + // Set wifi state to internet, to fake a connection from the NDM service. + // Remove once it is figured out how NDM uses AC to connect at console boot. + SetWifiLinkLevel(WifiLinkLevel::Best); + SetWifiState(WifiState::Internet); } u64 Handler::GetSystemTimeSince2000() const { @@ -136,10 +142,24 @@ void Handler::SetMacAddress(const MacAddress& addr) { std::memcpy(shared_page.wifi_macaddr, addr.data(), sizeof(MacAddress)); } +MacAddress Handler::GetMacAddress() { + MacAddress addr; + std::memcpy(addr.data(), shared_page.wifi_macaddr, sizeof(MacAddress)); + return addr; +} + void Handler::SetWifiLinkLevel(WifiLinkLevel level) { shared_page.wifi_link_level = static_cast(level); } +WifiLinkLevel Handler::GetWifiLinkLevel() { + return static_cast(shared_page.wifi_link_level); +} + +void Handler::SetWifiState(WifiState state) { + shared_page.wifi_state = static_cast(state); +} + void Handler::Set3DLed(u8 state) { shared_page.ledstate_3d = state; } diff --git a/src/core/hle/kernel/shared_page.h b/src/core/hle/kernel/shared_page.h index eebda76178..7c163bdd31 100644 --- a/src/core/hle/kernel/shared_page.h +++ b/src/core/hle/kernel/shared_page.h @@ -53,10 +53,20 @@ using MacAddress = std::array; constexpr MacAddress DefaultMac = {0x40, 0xF4, 0x07, 0x00, 0x00, 0x00}; enum class WifiLinkLevel : u8 { - OFF = 0, - POOR = 1, - GOOD = 2, - BEST = 3, + Off = 0, + Poor = 1, + Good = 2, + Best = 3, +}; + +enum class WifiState : u8 { + Invalid = 0, + Enabled = 1, + Internet = 2, + Local1 = 3, + Local2 = 4, + Local3 = 6, + Disabled = 7, }; struct SharedPageDef { @@ -70,7 +80,7 @@ struct SharedPageDef { DateTime date_time_1; // 40 u8 wifi_macaddr[6]; // 60 u8 wifi_link_level; // 66 - u8 wifi_unknown2; // 67 + u8 wifi_state; // 67 INSERT_PADDING_BYTES(0x80 - 0x68); // 68 float_le sliderstate_3d; // 80 u8 ledstate_3d; // 84 @@ -90,8 +100,14 @@ public: void SetMacAddress(const MacAddress&); + MacAddress GetMacAddress(); + void SetWifiLinkLevel(WifiLinkLevel); + WifiLinkLevel GetWifiLinkLevel(); + + void SetWifiState(WifiState); + void Set3DLed(u8); void Set3DSlider(float); diff --git a/src/core/hle/service/ac/ac.cpp b/src/core/hle/service/ac/ac.cpp index a719030283..3c569ed0c5 100644 --- a/src/core/hle/service/ac/ac.cpp +++ b/src/core/hle/service/ac/ac.cpp @@ -13,6 +13,7 @@ #include "core/hle/kernel/event.h" #include "core/hle/kernel/handle_table.h" #include "core/hle/kernel/resource_limit.h" +#include "core/hle/kernel/shared_page.h" #include "core/hle/result.h" #include "core/hle/service/ac/ac.h" #include "core/hle/service/ac/ac_i.h" @@ -40,76 +41,143 @@ void Module::Interface::CreateDefaultConfig(Kernel::HLERequestContext& ctx) { void Module::Interface::ConnectAsync(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); - rp.Skip(2, false); // ProcessId descriptor + const u32 pid = rp.PopPID(); ac->connect_event = rp.PopObject(); rp.Skip(2, false); // Buffer descriptor - if (ac->connect_event) { - ac->connect_event->SetName("AC:connect_event"); - ac->connect_event->Signal(); - ac->ac_connected = true; - } + ac->Connect(pid); IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); rb.Push(ResultSuccess); - LOG_WARNING(Service_AC, "(STUBBED) called"); + LOG_WARNING(Service_AC, "(STUBBED) called, pid={}", pid); } void Module::Interface::GetConnectResult(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); - rp.Skip(2, false); // ProcessId descriptor + [[maybe_unused]] const u32 pid = rp.PopPID(); IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); - rb.Push(ResultSuccess); + rb.Push(ac->connect_result); +} + +void Module::Interface::CancelConnectAsync(Kernel::HLERequestContext& ctx) { + IPC::RequestParser rp(ctx); + const u32 pid = rp.PopPID(); + + IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); + rb.Push(ac->ac_connected ? ErrorAlreadyConnected : ErrorNotConnected); + + LOG_WARNING(Service_AC, "(STUBBED) called, pid={}", pid); } void Module::Interface::CloseAsync(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); - rp.Skip(2, false); // ProcessId descriptor + const u32 pid = rp.PopPID(); ac->close_event = rp.PopObject(); - if (ac->ac_connected && ac->disconnect_event) { - ac->disconnect_event->Signal(); - } - - if (ac->close_event) { - ac->close_event->SetName("AC:close_event"); - ac->close_event->Signal(); - } - - ac->ac_connected = false; + ac->Disconnect(pid); IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); rb.Push(ResultSuccess); + + LOG_WARNING(Service_AC, "(STUBBED) called, pid={}", pid); } void Module::Interface::GetCloseResult(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); - rp.Skip(2, false); // ProcessId descriptor + [[maybe_unused]] const u32 pid = rp.PopPID(); IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); - rb.Push(ResultSuccess); - - LOG_WARNING(Service_AC, "(STUBBED) called"); + rb.Push(ac->close_result); } void Module::Interface::GetWifiStatus(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); - bool can_reach_internet = false; - std::shared_ptr socu_module = SOC::GetService(ac->system); - if (socu_module) { - can_reach_internet = socu_module->GetDefaultInterfaceInfo().has_value(); + if (!ac->ac_connected) { + IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); + rb.Push(ErrorNotConnected); + return; } IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); rb.Push(ResultSuccess); - rb.Push(static_cast(can_reach_internet ? (Settings::values.is_new_3ds - ? WifiStatus::STATUS_CONNECTED_N3DS - : WifiStatus::STATUS_CONNECTED_O3DS) - : WifiStatus::STATUS_DISCONNECTED)); + rb.Push(static_cast(WifiStatus::STATUS_CONNECTED_SLOT1)); + + LOG_WARNING(Service_AC, "(STUBBED) called"); +} + +void Module::Interface::GetCurrentAPInfo(Kernel::HLERequestContext& ctx) { + IPC::RequestParser rp(ctx); + const u32 len = rp.Pop(); + const u32 pid = rp.PopPID(); + + if (!ac->ac_connected) { + IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); + rb.Push(ErrorNotConnected); + return; + } + + constexpr const char* citra_ap = "Citra_AP"; + constexpr s16 good_signal_strength = 60; + constexpr u8 unknown1_value = 6; + constexpr u8 unknown2_value = 5; + constexpr u8 unknown3_value = 5; + constexpr u8 unknown4_value = 0; + + SharedPage::Handler& shared_page = ac->system.Kernel().GetSharedPageHandler(); + SharedPage::MacAddress mac = shared_page.GetMacAddress(); + + APInfo info{ + .ssid_len = static_cast(std::strlen(citra_ap)), + .bssid = mac, + .padding = 0, + .signal_strength = good_signal_strength, + .link_level = static_cast(shared_page.GetWifiLinkLevel()), + .unknown1 = unknown1_value, + .unknown2 = unknown2_value, + .unknown3 = unknown3_value, + .unknown4 = unknown4_value, + }; + std::strncpy(info.ssid.data(), citra_ap, info.ssid.size()); + + std::vector out_info(len); + std::memcpy(out_info.data(), &info, std::min(len, static_cast(sizeof(info)))); + + IPC::RequestBuilder rb = rp.MakeBuilder(1, 2); + rb.Push(ResultSuccess); + rb.PushStaticBuffer(out_info, 0); + + LOG_WARNING(Service_AC, "(STUBBED) called, pid={}", pid); +} + +void Module::Interface::GetConnectingInfraPriority(Kernel::HLERequestContext& ctx) { + IPC::RequestParser rp(ctx); + + if (!ac->ac_connected) { + IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); + rb.Push(ErrorNotConnected); + return; + } + + IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); + rb.Push(ResultSuccess); + rb.Push(static_cast(InfraPriority::PRIORITY_HIGH)); + + LOG_WARNING(Service_AC, "(STUBBED) called"); +} + +void Module::Interface::GetStatus(Kernel::HLERequestContext& ctx) { + IPC::RequestParser rp(ctx); + + IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); + rb.Push(ResultSuccess); + rb.Push(static_cast(ac->ac_connected ? NetworkStatus::STATUS_INTERNET + : NetworkStatus::STATUS_DISCONNECTED)); + + LOG_WARNING(Service_AC, "(STUBBED) called"); } void Module::Interface::GetInfraPriority(Kernel::HLERequestContext& ctx) { @@ -118,16 +186,28 @@ void Module::Interface::GetInfraPriority(Kernel::HLERequestContext& ctx) { IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); rb.Push(ResultSuccess); - rb.Push(0); // Infra Priority, default 0 + rb.Push(static_cast(InfraPriority::PRIORITY_HIGH)); LOG_WARNING(Service_AC, "(STUBBED) called"); } +void Module::Interface::SetFromApplication(Kernel::HLERequestContext& ctx) { + IPC::RequestParser rp(ctx); + const u32 unknown = rp.Pop(); + auto config = rp.PopStaticBuffer(); + + IPC::RequestBuilder rb = rp.MakeBuilder(1, 2); + rb.Push(ResultSuccess); + rb.PushStaticBuffer(config, 0); + + LOG_WARNING(Service_AC, "(STUBBED) called, unknown={}", unknown); +} + void Module::Interface::SetRequestEulaVersion(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); - u32 major = rp.Pop(); - u32 minor = rp.Pop(); + const u32 major = rp.Pop(); + const u32 minor = rp.Pop(); const std::vector& ac_config = rp.PopStaticBuffer(); @@ -140,6 +220,19 @@ void Module::Interface::SetRequestEulaVersion(Kernel::HLERequestContext& ctx) { LOG_WARNING(Service_AC, "(STUBBED) called, major={}, minor={}", major, minor); } +void Module::Interface::GetNZoneBeaconNotFoundEvent(Kernel::HLERequestContext& ctx) { + IPC::RequestParser rp(ctx); + rp.PopPID(); + auto event = rp.PopObject(); + + event->Signal(); + + IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); + rb.Push(ResultSuccess); + + LOG_WARNING(Service_AC, "(STUBBED) called"); +} + void Module::Interface::RegisterDisconnectEvent(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); rp.Skip(2, false); // ProcessId descriptor @@ -168,30 +261,93 @@ void Module::Interface::GetConnectingProxyEnable(Kernel::HLERequestContext& ctx) void Module::Interface::IsConnected(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); - u32 unk = rp.Pop(); - u32 unk_descriptor = rp.Pop(); - u32 unk_param = rp.Pop(); + const u32 unk = rp.Pop(); + const u32 pid = rp.PopPID(); IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); rb.Push(ResultSuccess); rb.Push(ac->ac_connected); - LOG_WARNING(Service_AC, "(STUBBED) called unk=0x{:08X} descriptor=0x{:08X} param=0x{:08X}", unk, - unk_descriptor, unk_param); + LOG_DEBUG(Service_AC, "(STUBBED) called unk=0x{:08X} pid={}", unk, pid); } void Module::Interface::SetClientVersion(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); - u32 version = rp.Pop(); - rp.Skip(2, false); // ProcessId descriptor + const u32 version = rp.Pop(); + rp.PopPID(); - LOG_WARNING(Service_AC, "(STUBBED) called, version: 0x{:08X}", version); + LOG_DEBUG(Service_AC, "(STUBBED) called, version: 0x{:08X}", version); IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); rb.Push(ResultSuccess); } +void Module::Connect(u32 pid) { + if (connect_event) { + connect_event->SetName("AC:connect_event"); + connect_event->Signal(); + } + + if (connected_pids.size() == 0) { + // TODO(PabloMK7) Publish to subscriber 0x300 + + ac_connected = true; + + // TODO(PabloMK7) Move shared page modification to NWM once it is implemented. + SharedPage::Handler& shared_page = system.Kernel().GetSharedPageHandler(); + const bool can_access_internet = CanAccessInternet(); + if (can_access_internet) { + shared_page.SetWifiState(SharedPage::WifiState::Internet); + shared_page.SetWifiLinkLevel(SharedPage::WifiLinkLevel::Best); + } else { + shared_page.SetWifiState(SharedPage::WifiState::Enabled); + shared_page.SetWifiLinkLevel(SharedPage::WifiLinkLevel::Off); + } + } + + if (connected_pids.find(pid) == connected_pids.end()) { + connected_pids.insert(pid); + connect_result = ResultSuccess; + } else { + connect_result = ErrorAlreadyConnected; + } +} + +void Module::Disconnect(u32 pid) { + if (close_event) { + close_event->SetName("AC:close_event"); + close_event->Signal(); + } + + if (connected_pids.find(pid) != connected_pids.end()) { + connected_pids.erase(pid); + close_result = ResultSuccess; + } else { + close_result = ErrorNotConnected; + } + + if (connected_pids.size() == 0) { + ac_connected = false; + if (disconnect_event) { + disconnect_event->Signal(); + } + + // TODO(PabloMK7) Move shared page modification to NWM once it is implemented. + SharedPage::Handler& shared_page = system.Kernel().GetSharedPageHandler(); + shared_page.SetWifiState(SharedPage::WifiState::Enabled); + shared_page.SetWifiLinkLevel(SharedPage::WifiLinkLevel::Off); + } +} + +bool Module::CanAccessInternet() { + std::shared_ptr socu_module = SOC::GetService(system); + if (socu_module) { + return socu_module->GetDefaultInterfaceInfo().has_value(); + } + return false; +} + Module::Interface::Interface(std::shared_ptr ac, const char* name, u32 max_session) : ServiceFramework(name, max_session), ac(std::move(ac)) {} @@ -202,6 +358,10 @@ void InstallInterfaces(Core::System& system) { std::make_shared(ac)->InstallAsService(service_manager); } +std::shared_ptr GetService(Core::System& system) { + return system.ServiceManager().GetService("ac:u"); +} + Module::Module(Core::System& system_) : system(system_) {} template @@ -210,6 +370,13 @@ void Module::serialize(Archive& ar, const unsigned int) { ar& close_event; ar& connect_event; ar& disconnect_event; + u32 connect_result_32 = connect_result.raw; + ar& connect_result_32; + connect_result.raw = connect_result_32; + u32 close_result_32 = close_result.raw; + ar& close_result_32; + close_result.raw = close_result_32; + ar& connected_pids; // default_config is never written to } SERIALIZE_IMPL(Module) diff --git a/src/core/hle/service/ac/ac.h b/src/core/hle/service/ac/ac.h index 2b0d2e5cef..4fdd8e4bc1 100644 --- a/src/core/hle/service/ac/ac.h +++ b/src/core/hle/service/ac/ac.h @@ -17,6 +17,7 @@ class Event; } namespace Service::AC { +class AC_U; class Module final { public: explicit Module(Core::System& system_); @@ -58,6 +59,15 @@ public: */ void GetConnectResult(Kernel::HLERequestContext& ctx); + /** + * AC::CancelConnectAsync service function + * Inputs: + * 1 : ProcessId Header + * Outputs: + * 1 : Result of function, 0 on success, otherwise error code + */ + void CancelConnectAsync(Kernel::HLERequestContext& ctx); + /** * AC::CloseAsync service function * Inputs: @@ -78,14 +88,40 @@ public: */ void GetCloseResult(Kernel::HLERequestContext& ctx); + /** + * AC::GetStatus service function + * Outputs: + * 1 : Result of function, 0 on success, otherwise error code + * 2 : Output status + */ + void GetStatus(Kernel::HLERequestContext& ctx); + /** * AC::GetWifiStatus service function * Outputs: * 1 : Result of function, 0 on success, otherwise error code - * 2 : Output connection type, 0 = none, 1 = Old3DS Internet, 2 = New3DS Internet. + * 2 : Output wifi status */ void GetWifiStatus(Kernel::HLERequestContext& ctx); + /** + * AC::GetCurrentAPInfo service function + * Inputs: + * 1 : Size + * 2-3 : ProcessID + * Outputs: + * 1 : Result of function, 0 on success, otherwise error code + */ + void GetCurrentAPInfo(Kernel::HLERequestContext& ctx); + + /** + * AC::GetConnectingInfraPriority service function + * Outputs: + * 1 : Result of function, 0 on success, otherwise error code + * 2 : Output connecting priority + */ + void GetConnectingInfraPriority(Kernel::HLERequestContext& ctx); + /** * AC::GetInfraPriority service function * Inputs: @@ -97,6 +133,15 @@ public: */ void GetInfraPriority(Kernel::HLERequestContext& ctx); + /** + * AC::SetFromApplication service function + * Inputs: + * 1-2 : Input config + * Outputs: + * 1-2 : Output config + */ + void SetFromApplication(Kernel::HLERequestContext& ctx); + /** * AC::SetRequestEulaVersion service function * Inputs: @@ -112,6 +157,17 @@ public: */ void SetRequestEulaVersion(Kernel::HLERequestContext& ctx); + /** + * AC::GetNZoneBeaconNotFoundEvent service function + * Inputs: + * 1 : ProcessId Header + * 3 : Copy Handle Header + * 4 : Event handle, should be signaled when AC cannot find NZone + * Outputs: + * 1 : Result of function, 0 on success, otherwise error code + */ + void GetNZoneBeaconNotFoundEvent(Kernel::HLERequestContext& ctx); + /** * AC::RegisterDisconnectEvent service function * Inputs: @@ -153,12 +209,46 @@ public: }; protected: + static constexpr Result ErrorNotConnected = + Result(302, ErrorModule::AC, ErrorSummary::InvalidState, ErrorLevel::Usage); + + static constexpr Result ErrorAlreadyConnected = + Result(301, ErrorModule::AC, ErrorSummary::InvalidState, ErrorLevel::Usage); + + enum class NetworkStatus { + STATUS_DISCONNECTED = 0, + STATUS_ENABLED = 1, + STATUS_LOCAL = 2, + STATUS_INTERNET = 3, + }; + enum class WifiStatus { STATUS_DISCONNECTED = 0, - STATUS_CONNECTED_O3DS = 1, - STATUS_CONNECTED_N3DS = 2, + STATUS_CONNECTED_SLOT1 = (1 << 0), + STATUS_CONNECTED_SLOT2 = (1 << 1), + STATUS_CONNECTED_SLOT3 = (1 << 2), }; + enum class InfraPriority { + PRIORITY_HIGH = 0, + PRIORITY_LOW = 1, + PRIORITY_NONE = 2, + }; + + struct APInfo { + u32 ssid_len; + std::array ssid; + std::array bssid; + u16 padding; + s16 signal_strength; + u8 link_level; + u8 unknown1; + u8 unknown2; + u8 unknown3; + u16 unknown4; + }; + static_assert(sizeof(APInfo) == 0x34, "Invalid APInfo size"); + struct ACConfig { std::array data; }; @@ -170,6 +260,15 @@ protected: std::shared_ptr close_event; std::shared_ptr connect_event; std::shared_ptr disconnect_event; + Result connect_result = ResultSuccess; + Result close_result = ResultSuccess; + std::set connected_pids; + + void Connect(u32 pid); + + void Disconnect(u32 pid); + + bool CanAccessInternet(); private: Core::System& system; @@ -181,6 +280,8 @@ private: void InstallInterfaces(Core::System& system); +std::shared_ptr GetService(Core::System& system); + } // namespace Service::AC BOOST_CLASS_EXPORT_KEY(Service::AC::Module) diff --git a/src/core/hle/service/ac/ac_i.cpp b/src/core/hle/service/ac/ac_i.cpp index f7766b3249..ab361a0293 100644 --- a/src/core/hle/service/ac/ac_i.cpp +++ b/src/core/hle/service/ac/ac_i.cpp @@ -13,19 +13,22 @@ AC_I::AC_I(std::shared_ptr ac) : Module::Interface(std::move(ac), "ac:i" {0x0001, &AC_I::CreateDefaultConfig, "CreateDefaultConfig"}, {0x0004, &AC_I::ConnectAsync, "ConnectAsync"}, {0x0005, &AC_I::GetConnectResult, "GetConnectResult"}, - {0x0007, nullptr, "CancelConnectAsync"}, + {0x0007, &AC_I::CancelConnectAsync, "CancelConnectAsync"}, {0x0008, &AC_I::CloseAsync, "CloseAsync"}, {0x0009, &AC_I::GetCloseResult, "GetCloseResult"}, {0x000A, nullptr, "GetLastErrorCode"}, - {0x000C, nullptr, "GetStatus"}, + {0x000C, &AC_I::GetStatus, "GetStatus"}, {0x000D, &AC_I::GetWifiStatus, "GetWifiStatus"}, - {0x000E, nullptr, "GetCurrentAPInfo"}, + {0x000E, &AC_I::GetCurrentAPInfo, "GetCurrentAPInfo"}, + {0x000F, &AC_I::GetConnectingInfraPriority, "GetConnectingInfraPriority"}, {0x0010, nullptr, "GetCurrentNZoneInfo"}, {0x0011, nullptr, "GetNZoneApNumService"}, {0x001D, nullptr, "ScanAPs"}, {0x0024, nullptr, "AddDenyApType"}, {0x0027, &AC_I::GetInfraPriority, "GetInfraPriority"}, + {0x002C, &AC_I::SetFromApplication, "SetFromApplication"}, {0x002D, &AC_I::SetRequestEulaVersion, "SetRequestEulaVersion"}, + {0x002F, &AC_I::GetNZoneBeaconNotFoundEvent, "GetNZoneBeaconNotFoundEvent"}, {0x0030, &AC_I::RegisterDisconnectEvent, "RegisterDisconnectEvent"}, {0x0036, &AC_I::GetConnectingProxyEnable, "GetConnectingProxyEnable"}, {0x003C, nullptr, "GetAPSSIDList"}, diff --git a/src/core/hle/service/ac/ac_u.cpp b/src/core/hle/service/ac/ac_u.cpp index 12506962c5..bc2af00282 100644 --- a/src/core/hle/service/ac/ac_u.cpp +++ b/src/core/hle/service/ac/ac_u.cpp @@ -13,19 +13,22 @@ AC_U::AC_U(std::shared_ptr ac) : Module::Interface(std::move(ac), "ac:u" {0x0001, &AC_U::CreateDefaultConfig, "CreateDefaultConfig"}, {0x0004, &AC_U::ConnectAsync, "ConnectAsync"}, {0x0005, &AC_U::GetConnectResult, "GetConnectResult"}, - {0x0007, nullptr, "CancelConnectAsync"}, + {0x0007, &AC_U::CancelConnectAsync, "CancelConnectAsync"}, {0x0008, &AC_U::CloseAsync, "CloseAsync"}, {0x0009, &AC_U::GetCloseResult, "GetCloseResult"}, {0x000A, nullptr, "GetLastErrorCode"}, - {0x000C, nullptr, "GetStatus"}, + {0x000C, &AC_U::GetStatus, "GetStatus"}, {0x000D, &AC_U::GetWifiStatus, "GetWifiStatus"}, - {0x000E, nullptr, "GetCurrentAPInfo"}, + {0x000E, &AC_U::GetCurrentAPInfo, "GetCurrentAPInfo"}, + {0x000F, &AC_U::GetConnectingInfraPriority, "GetConnectingInfraPriority"}, {0x0010, nullptr, "GetCurrentNZoneInfo"}, {0x0011, nullptr, "GetNZoneApNumService"}, {0x001D, nullptr, "ScanAPs"}, {0x0024, nullptr, "AddDenyApType"}, {0x0027, &AC_U::GetInfraPriority, "GetInfraPriority"}, + {0x002C, &AC_U::SetFromApplication, "SetFromApplication"}, {0x002D, &AC_U::SetRequestEulaVersion, "SetRequestEulaVersion"}, + {0x002F, &AC_U::GetNZoneBeaconNotFoundEvent, "GetNZoneBeaconNotFoundEvent"}, {0x0030, &AC_U::RegisterDisconnectEvent, "RegisterDisconnectEvent"}, {0x0036, &AC_U::GetConnectingProxyEnable, "GetConnectingProxyEnable"}, {0x003C, nullptr, "GetAPSSIDList"}, diff --git a/src/core/hle/service/nwm/nwm_uds.cpp b/src/core/hle/service/nwm/nwm_uds.cpp index d8662ccbe8..fcd42d991f 100644 --- a/src/core/hle/service/nwm/nwm_uds.cpp +++ b/src/core/hle/service/nwm/nwm_uds.cpp @@ -575,6 +575,10 @@ void NWM_UDS::Shutdown(Kernel::HLERequestContext& ctx) { recv_buffer_memory.reset(); + SharedPage::Handler& shared_page = system.Kernel().GetSharedPageHandler(); + shared_page.SetWifiLinkLevel(SharedPage::WifiLinkLevel::Off); + shared_page.SetWifiState(SharedPage::WifiState::Enabled); + IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); rb.Push(ResultSuccess); LOG_DEBUG(Service_NWM, "called"); @@ -667,6 +671,10 @@ ResultVal> NWM_UDS::Initialize( channel_data.clear(); } + SharedPage::Handler& shared_page = system.Kernel().GetSharedPageHandler(); + shared_page.SetWifiLinkLevel(SharedPage::WifiLinkLevel::Best); + shared_page.SetWifiState(SharedPage::WifiState::Local1); + return connection_status_event; } @@ -1506,7 +1514,6 @@ NWM_UDS::NWM_UDS(Core::System& system) : ServiceFramework("nwm::UDS"), system(sy } system.Kernel().GetSharedPageHandler().SetMacAddress(mac); - system.Kernel().GetSharedPageHandler().SetWifiLinkLevel(SharedPage::WifiLinkLevel::BEST); if (auto room_member = Network::GetRoomMember().lock()) { wifi_packet_received = room_member->BindOnWifiPacketReceived(