// Copyright 2016 Citra Emulator Project // Licensed under GPLv2 or any later version // Refer to the license.txt file included. #include "common/common_types.h" #include "common/logging/log.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/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/memory.h" namespace Service { namespace AC { void Module::Interface::CreateDefaultConfig(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx, 0x1, 0, 0); std::size_t desc_size; VAddr ac_config_addr = rp.PeekStaticBuffer(0, &desc_size); ASSERT_MSG(desc_size >= sizeof(Module::ACConfig), "Output buffer size can't fit ACConfig structure"); Memory::WriteBlock(ac_config_addr, &ac->default_config, sizeof(ACConfig)); IPC::RequestBuilder rb = rp.MakeBuilder(1, 2); rb.Push(RESULT_SUCCESS); rb.PushStaticBuffer(ac_config_addr, sizeof(ACConfig), 0); LOG_WARNING(Service_AC, "(STUBBED) called"); } void Module::Interface::ConnectAsync(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx, 0x4, 0, 6); rp.Skip(2, false); // ProcessId descriptor ac->connect_event = rp.PopObject(); if (ac->connect_event) { ac->connect_event->name = "AC:connect_event"; ac->connect_event->Signal(); ac->ac_connected = true; } IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); rb.Push(RESULT_SUCCESS); LOG_WARNING(Service_AC, "(STUBBED) called"); } void Module::Interface::GetConnectResult(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx, 0x5, 0, 2); rp.Skip(2, false); // ProcessId descriptor IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); rb.Push(RESULT_SUCCESS); } void Module::Interface::CloseAsync(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx, 0x8, 0, 4); rp.Skip(2, false); // ProcessId descriptor ac->close_event = rp.PopObject(); if (ac->ac_connected && ac->disconnect_event) { ac->disconnect_event->Signal(); } if (ac->close_event) { ac->close_event->name = "AC:close_event"; ac->close_event->Signal(); } ac->ac_connected = false; IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); rb.Push(RESULT_SUCCESS); } void Module::Interface::GetCloseResult(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx, 0x9, 0, 2); rp.Skip(2, false); // ProcessId descriptor IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); rb.Push(RESULT_SUCCESS); LOG_WARNING(Service_AC, "(STUBBED) called"); } void Module::Interface::GetWifiStatus(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx, 0xD, 0, 0); // TODO(purpasmart96): This function is only a stub, // it returns a valid result without implementing full functionality. IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); rb.Push(RESULT_SUCCESS); rb.Push(0); // Connection type set to none LOG_WARNING(Service_AC, "(STUBBED) called"); } void Module::Interface::GetInfraPriority(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx, 0x27, 0, 2); VAddr ac_config = rp.PopStaticBuffer(); IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); rb.Push(RESULT_SUCCESS); rb.Push(0); // Infra Priority, default 0 LOG_WARNING(Service_AC, "(STUBBED) called"); } void Module::Interface::SetRequestEulaVersion(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx, 0x2D, 2, 2); u32 major = rp.Pop(); u32 minor = rp.Pop(); VAddr ac_config = rp.PopStaticBuffer(); // TODO(Subv): Copy over the input ACConfig to the stored ACConfig. IPC::RequestBuilder rb = rp.MakeBuilder(1, 2); rb.Push(RESULT_SUCCESS); rb.PushStaticBuffer(ac_config, sizeof(ACConfig), 0); LOG_WARNING(Service_AC, "(STUBBED) called, major=%u, minor=%u", major, minor); } void Module::Interface::RegisterDisconnectEvent(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx, 0x30, 0, 4); rp.Skip(2, false); // ProcessId descriptor ac->disconnect_event = rp.PopObject(); if (ac->disconnect_event) { ac->disconnect_event->name = "AC:disconnect_event"; } IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); rb.Push(RESULT_SUCCESS); LOG_WARNING(Service_AC, "(STUBBED) called"); } void Module::Interface::IsConnected(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx, 0x3E, 1, 2); u32 unk = rp.Pop(); u32 unk_descriptor = rp.Pop(); u32 unk_param = rp.Pop(); IPC::RequestBuilder rb = rp.MakeBuilder(2, 0); rb.Push(RESULT_SUCCESS); rb.Push(ac->ac_connected); LOG_WARNING(Service_AC, "(STUBBED) called unk=%08X descriptor=%08X param=%08X", unk, unk_descriptor, unk_param); } void Module::Interface::SetClientVersion(Kernel::HLERequestContext& ctx) { IPC::RequestParser rp(ctx, 0x40, 1, 2); u32 version = rp.Pop(); rp.Skip(2, false); // ProcessId descriptor LOG_WARNING(Service_AC, "(STUBBED) called, version: 0x%08X", version); IPC::RequestBuilder rb = rp.MakeBuilder(1, 0); rb.Push(RESULT_SUCCESS); } Module::Interface::Interface(std::shared_ptr ac, const char* name, u32 max_session) : ac(std::move(ac)), ServiceFramework(name, max_session) {} void InstallInterfaces(SM::ServiceManager& service_manager) { auto ac = std::make_shared(); std::make_shared(ac)->InstallAsService(service_manager); std::make_shared(ac)->InstallAsService(service_manager); } } // namespace AC } // namespace Service