// Copyright 2016 Citra Emulator Project // Licensed under GPLv2 or any later version // Refer to the license.txt file included. #include #include #include "common/archives.h" #include "common/common_types.h" #include "common/logging/log.h" #include "common/settings.h" #include "core/core.h" #include "core/hle/ipc.h" #include "core/hle/ipc_helpers.h" #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" #include "core/hle/service/ac/ac_u.h" #include "core/hle/service/nwm/nwm_inf.h" #include "core/hle/service/soc/soc_u.h" #include "network/network.h" #include "network/room.h" #include "core/memory.h" SERIALIZE_EXPORT_IMPL(Service::AC::Module) SERVICE_CONSTRUCT_IMPL(Service::AC::Module) namespace Service::AC { void Module::Interface::CreateDefaultConfig(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); std::vector buffer(sizeof(ACConfig)); std::memcpy(buffer.data(), &ac->default_config, buffer.size()); IPC::RequestBuilder rb = rp.MakeBuilder(1, 2); rb.Push(ResultSuccess); rb.PushStaticBuffer(std::move(buffer), 0); LOG_WARNING(Service_AC, "(STUBBED) called"); } void Module::Interface::ConnectAsync(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); const u32 pid = rp.PopPID(); ac->connect_event = rp.PopObject(); rp.Skip(2, false); // Buffer descriptor ac->Connect(pid); IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); rb.Push(ResultSuccess); LOG_WARNING(Service_AC, "(STUBBED) called, pid={}", pid); } void Module::Interface::GetConnectResult(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); [[maybe_unused]] const u32 pid = rp.PopPID(); IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); 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); const u32 pid = rp.PopPID(); ac->close_event = rp.PopObject(); 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); [[maybe_unused]] const u32 pid = rp.PopPID(); IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); rb.Push(ac->close_result); } void Module::Interface::GetWifiStatus(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(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* lime3ds_ap = "Lime3DS_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(lime3ds_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(), lime3ds_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::ScanAPs(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); // I used 3dbrew.org for the information on the individual request inputs // Arg 0 is Header code, which is ignored // Arg 1 is Size const u32 size = rp.Pop(); LOG_WARNING(Service_AC, "Size: {}", size); // Arg 2 is CallingPID value (PID Header) // Arg 3 is PID const u32 pid = rp.PopPID(); LOG_WARNING(Service_AC, "PID: {}", pid); std::shared_ptr thread = ctx.ClientThread(); auto current_process = Core::System::GetInstance().GetCurrentProcess(); Memory::MemorySystem& memory = Core::System::GetInstance().Memory(); LOG_WARNING(Service_AC, "Retrieved thread, process and memory"); // According to 3dbrew, the output structure pointer is located 0x100 bytes after the beginning // of cmd_buff VAddr cmd_addr = thread->GetCommandBufferAddress(); VAddr buffer_vaddr = cmd_addr + 0x100; u32* buffer_info = static_cast(memory.GetPointer(buffer_vaddr)); const u32 descr = buffer_info[0]; ASSERT(descr == ((size << 14) | 2)); // preliminary check const VAddr output_buffer = buffer_info[1]; // address to output buffer Network::MacAddress mac = Network::BroadcastMac; u32 mac1 = (mac[0] << 8) | (mac[1]); u32 mac2 = (mac[2] << 24) | (mac[3] << 16) | (mac[4] << 8) | (mac[5]); std::array cmd_buf; cmd_buf[0] = 0x000603C4; cmd_buf[1] = size; cmd_buf[2] = 0; // dummy data cmd_buf[3] = 0; // dummy data cmd_buf[4] = mac1; cmd_buf[5] = mac2; cmd_buf[16] = 0; cmd_buf[17] = 0; // set to 0 to ignore it cmd_buf[18] = (size << 4) | 12; // should be considered correct for mapped buffer cmd_buf[19] = output_buffer; LOG_WARNING(Service_AC, "Finished setting up command buffer"); auto context = std::make_shared(Core::System::GetInstance().Kernel(), ctx.Session(), thread); LOG_WARNING(Service_AC, "Created context"); context->PopulateFromIncomingCommandBuffer(cmd_buf.data(), current_process); LOG_WARNING(Service_AC, "Finished setting up context"); auto nwm_inf = Core::System::GetInstance().ServiceManager().GetService("nwm::INF"); LOG_WARNING(Service_AC, "Calling NWM_INF::RecvBeaconBroadcastData"); nwm_inf->HandleSyncRequest(*context); LOG_WARNING(Service_AC, "Returned to AC::ScanAPs"); // Response should be // 0: Header Code (ignored) // 1: Result Code (Success/Unknown/etc.) // 2: ¿Parsed? beacon data IPC::RequestBuilder rb = rp.MakeBuilder(1, 2); IPC::RequestParser rp2(*context); rb.Push(rp2.Pop()); Kernel::MappedBuffer mapped_buffer = rp2.PopMappedBuffer(); rb.PushMappedBuffer(mapped_buffer); LOG_WARNING(Service_AC, "(STUBBED) called, pid={}", pid); } void Module::Interface::GetInfraPriority(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); [[maybe_unused]] const std::vector& ac_config = rp.PopStaticBuffer(); 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::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); const u32 major = rp.Pop(); const u32 minor = rp.Pop(); const std::vector& ac_config = rp.PopStaticBuffer(); // TODO(Subv): Copy over the input ACConfig to the stored ACConfig. IPC::RequestBuilder rb = rp.MakeBuilder(1, 2); rb.Push(ResultSuccess); rb.PushStaticBuffer(std::move(ac_config), 0); 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 ac->disconnect_event = rp.PopObject(); if (ac->disconnect_event) { ac->disconnect_event->SetName("AC:disconnect_event"); } IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); rb.Push(ResultSuccess); LOG_WARNING(Service_AC, "(STUBBED) called"); } void Module::Interface::GetConnectingProxyEnable(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); constexpr bool proxy_enabled = false; IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); rb.Push(ResultSuccess); rb.Push(proxy_enabled); LOG_WARNING(Service_AC, "(STUBBED) called"); } void Module::Interface::IsConnected(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); 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_DEBUG(Service_AC, "(STUBBED) called unk=0x{:08X} pid={}", unk, pid); } void Module::Interface::SetClientVersion(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx); const u32 version = rp.Pop(); rp.PopPID(); 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)) {} void InstallInterfaces(Core::System& system) { auto& service_manager = system.ServiceManager(); auto ac = std::make_shared(system); std::make_shared(ac)->InstallAsService(service_manager); 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 void Module::serialize(Archive& ar, const unsigned int) { ar & ac_connected; 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) } // namespace Service::AC