mirror of
https://github.com/starr-dusT/yuzu-mainline
synced 2024-03-05 21:12:25 -08:00
742 lines
27 KiB
C++
742 lines
27 KiB
C++
// SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#include <memory>
|
|
|
|
#include "core/core.h"
|
|
#include "core/hle/service/ldn/lan_discovery.h"
|
|
#include "core/hle/service/ldn/ldn.h"
|
|
#include "core/hle/service/ldn/ldn_results.h"
|
|
#include "core/hle/service/ldn/ldn_types.h"
|
|
#include "core/internal_network/network.h"
|
|
#include "core/internal_network/network_interface.h"
|
|
#include "network/network.h"
|
|
|
|
// This is defined by synchapi.h and conflicts with ServiceContext::CreateEvent
|
|
#undef CreateEvent
|
|
|
|
namespace Service::LDN {
|
|
|
|
class IMonitorService final : public ServiceFramework<IMonitorService> {
|
|
public:
|
|
explicit IMonitorService(Core::System& system_) : ServiceFramework{system_, "IMonitorService"} {
|
|
// clang-format off
|
|
static const FunctionInfo functions[] = {
|
|
{0, nullptr, "GetStateForMonitor"},
|
|
{1, nullptr, "GetNetworkInfoForMonitor"},
|
|
{2, nullptr, "GetIpv4AddressForMonitor"},
|
|
{3, nullptr, "GetDisconnectReasonForMonitor"},
|
|
{4, nullptr, "GetSecurityParameterForMonitor"},
|
|
{5, nullptr, "GetNetworkConfigForMonitor"},
|
|
{100, nullptr, "InitializeMonitor"},
|
|
{101, nullptr, "FinalizeMonitor"},
|
|
};
|
|
// clang-format on
|
|
|
|
RegisterHandlers(functions);
|
|
}
|
|
};
|
|
|
|
class LDNM final : public ServiceFramework<LDNM> {
|
|
public:
|
|
explicit LDNM(Core::System& system_) : ServiceFramework{system_, "ldn:m"} {
|
|
// clang-format off
|
|
static const FunctionInfo functions[] = {
|
|
{0, &LDNM::CreateMonitorService, "CreateMonitorService"}
|
|
};
|
|
// clang-format on
|
|
|
|
RegisterHandlers(functions);
|
|
}
|
|
|
|
void CreateMonitorService(Kernel::HLERequestContext& ctx) {
|
|
LOG_DEBUG(Service_LDN, "called");
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
|
rb.Push(ResultSuccess);
|
|
rb.PushIpcInterface<IMonitorService>(system);
|
|
}
|
|
};
|
|
|
|
class ISystemLocalCommunicationService final
|
|
: public ServiceFramework<ISystemLocalCommunicationService> {
|
|
public:
|
|
explicit ISystemLocalCommunicationService(Core::System& system_)
|
|
: ServiceFramework{system_, "ISystemLocalCommunicationService"} {
|
|
// clang-format off
|
|
static const FunctionInfo functions[] = {
|
|
{0, nullptr, "GetState"},
|
|
{1, nullptr, "GetNetworkInfo"},
|
|
{2, nullptr, "GetIpv4Address"},
|
|
{3, nullptr, "GetDisconnectReason"},
|
|
{4, nullptr, "GetSecurityParameter"},
|
|
{5, nullptr, "GetNetworkConfig"},
|
|
{100, nullptr, "AttachStateChangeEvent"},
|
|
{101, nullptr, "GetNetworkInfoLatestUpdate"},
|
|
{102, nullptr, "Scan"},
|
|
{103, nullptr, "ScanPrivate"},
|
|
{104, nullptr, "SetWirelessControllerRestriction"},
|
|
{200, nullptr, "OpenAccessPoint"},
|
|
{201, nullptr, "CloseAccessPoint"},
|
|
{202, nullptr, "CreateNetwork"},
|
|
{203, nullptr, "CreateNetworkPrivate"},
|
|
{204, nullptr, "DestroyNetwork"},
|
|
{205, nullptr, "Reject"},
|
|
{206, nullptr, "SetAdvertiseData"},
|
|
{207, nullptr, "SetStationAcceptPolicy"},
|
|
{208, nullptr, "AddAcceptFilterEntry"},
|
|
{209, nullptr, "ClearAcceptFilter"},
|
|
{300, nullptr, "OpenStation"},
|
|
{301, nullptr, "CloseStation"},
|
|
{302, nullptr, "Connect"},
|
|
{303, nullptr, "ConnectPrivate"},
|
|
{304, nullptr, "Disconnect"},
|
|
{400, nullptr, "InitializeSystem"},
|
|
{401, nullptr, "FinalizeSystem"},
|
|
{402, nullptr, "SetOperationMode"},
|
|
{403, nullptr, "InitializeSystem2"},
|
|
};
|
|
// clang-format on
|
|
|
|
RegisterHandlers(functions);
|
|
}
|
|
};
|
|
|
|
class IUserLocalCommunicationService final
|
|
: public ServiceFramework<IUserLocalCommunicationService> {
|
|
public:
|
|
explicit IUserLocalCommunicationService(Core::System& system_)
|
|
: ServiceFramework{system_, "IUserLocalCommunicationService", ServiceThreadType::CreateNew},
|
|
service_context{system, "IUserLocalCommunicationService"},
|
|
room_network{system_.GetRoomNetwork()}, lan_discovery{room_network} {
|
|
// clang-format off
|
|
static const FunctionInfo functions[] = {
|
|
{0, &IUserLocalCommunicationService::GetState, "GetState"},
|
|
{1, &IUserLocalCommunicationService::GetNetworkInfo, "GetNetworkInfo"},
|
|
{2, &IUserLocalCommunicationService::GetIpv4Address, "GetIpv4Address"},
|
|
{3, &IUserLocalCommunicationService::GetDisconnectReason, "GetDisconnectReason"},
|
|
{4, &IUserLocalCommunicationService::GetSecurityParameter, "GetSecurityParameter"},
|
|
{5, &IUserLocalCommunicationService::GetNetworkConfig, "GetNetworkConfig"},
|
|
{100, &IUserLocalCommunicationService::AttachStateChangeEvent, "AttachStateChangeEvent"},
|
|
{101, &IUserLocalCommunicationService::GetNetworkInfoLatestUpdate, "GetNetworkInfoLatestUpdate"},
|
|
{102, &IUserLocalCommunicationService::Scan, "Scan"},
|
|
{103, &IUserLocalCommunicationService::ScanPrivate, "ScanPrivate"},
|
|
{104, &IUserLocalCommunicationService::SetWirelessControllerRestriction, "SetWirelessControllerRestriction"},
|
|
{200, &IUserLocalCommunicationService::OpenAccessPoint, "OpenAccessPoint"},
|
|
{201, &IUserLocalCommunicationService::CloseAccessPoint, "CloseAccessPoint"},
|
|
{202, &IUserLocalCommunicationService::CreateNetwork, "CreateNetwork"},
|
|
{203, &IUserLocalCommunicationService::CreateNetworkPrivate, "CreateNetworkPrivate"},
|
|
{204, &IUserLocalCommunicationService::DestroyNetwork, "DestroyNetwork"},
|
|
{205, nullptr, "Reject"},
|
|
{206, &IUserLocalCommunicationService::SetAdvertiseData, "SetAdvertiseData"},
|
|
{207, &IUserLocalCommunicationService::SetStationAcceptPolicy, "SetStationAcceptPolicy"},
|
|
{208, &IUserLocalCommunicationService::AddAcceptFilterEntry, "AddAcceptFilterEntry"},
|
|
{209, nullptr, "ClearAcceptFilter"},
|
|
{300, &IUserLocalCommunicationService::OpenStation, "OpenStation"},
|
|
{301, &IUserLocalCommunicationService::CloseStation, "CloseStation"},
|
|
{302, &IUserLocalCommunicationService::Connect, "Connect"},
|
|
{303, nullptr, "ConnectPrivate"},
|
|
{304, &IUserLocalCommunicationService::Disconnect, "Disconnect"},
|
|
{400, &IUserLocalCommunicationService::Initialize, "Initialize"},
|
|
{401, &IUserLocalCommunicationService::Finalize, "Finalize"},
|
|
{402, &IUserLocalCommunicationService::Initialize2, "Initialize2"},
|
|
};
|
|
// clang-format on
|
|
|
|
RegisterHandlers(functions);
|
|
|
|
state_change_event =
|
|
service_context.CreateEvent("IUserLocalCommunicationService:StateChangeEvent");
|
|
}
|
|
|
|
~IUserLocalCommunicationService() {
|
|
if (is_initialized) {
|
|
if (auto room_member = room_network.GetRoomMember().lock()) {
|
|
room_member->Unbind(ldn_packet_received);
|
|
}
|
|
}
|
|
|
|
service_context.CloseEvent(state_change_event);
|
|
}
|
|
|
|
/// Callback to parse and handle a received LDN packet.
|
|
void OnLDNPacketReceived(const Network::LDNPacket& packet) {
|
|
lan_discovery.ReceivePacket(packet);
|
|
}
|
|
|
|
void OnEventFired() {
|
|
state_change_event->Signal();
|
|
}
|
|
|
|
void GetState(Kernel::HLERequestContext& ctx) {
|
|
State state = State::Error;
|
|
|
|
if (is_initialized) {
|
|
state = lan_discovery.GetState();
|
|
}
|
|
|
|
IPC::ResponseBuilder rb{ctx, 3};
|
|
rb.Push(ResultSuccess);
|
|
rb.PushEnum(state);
|
|
}
|
|
|
|
void GetNetworkInfo(Kernel::HLERequestContext& ctx) {
|
|
const auto write_buffer_size = ctx.GetWriteBufferSize();
|
|
|
|
if (write_buffer_size != sizeof(NetworkInfo)) {
|
|
LOG_ERROR(Service_LDN, "Invalid buffer size {}", write_buffer_size);
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(ResultBadInput);
|
|
return;
|
|
}
|
|
|
|
NetworkInfo network_info{};
|
|
const auto rc = lan_discovery.GetNetworkInfo(network_info);
|
|
if (rc.IsError()) {
|
|
LOG_ERROR(Service_LDN, "NetworkInfo is not valid {}", rc.raw);
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(rc);
|
|
return;
|
|
}
|
|
|
|
ctx.WriteBuffer<NetworkInfo>(network_info);
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(ResultSuccess);
|
|
}
|
|
|
|
void GetIpv4Address(Kernel::HLERequestContext& ctx) {
|
|
const auto network_interface = Network::GetSelectedNetworkInterface();
|
|
|
|
if (!network_interface) {
|
|
LOG_ERROR(Service_LDN, "No network interface available");
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(ResultNoIpAddress);
|
|
return;
|
|
}
|
|
|
|
Ipv4Address current_address{Network::TranslateIPv4(network_interface->ip_address)};
|
|
Ipv4Address subnet_mask{Network::TranslateIPv4(network_interface->subnet_mask)};
|
|
|
|
// When we're connected to a room, spoof the hosts IP address
|
|
if (auto room_member = room_network.GetRoomMember().lock()) {
|
|
if (room_member->IsConnected()) {
|
|
current_address = room_member->GetFakeIpAddress();
|
|
}
|
|
}
|
|
|
|
std::reverse(std::begin(current_address), std::end(current_address)); // ntohl
|
|
std::reverse(std::begin(subnet_mask), std::end(subnet_mask)); // ntohl
|
|
|
|
IPC::ResponseBuilder rb{ctx, 4};
|
|
rb.Push(ResultSuccess);
|
|
rb.PushRaw(current_address);
|
|
rb.PushRaw(subnet_mask);
|
|
}
|
|
|
|
void GetDisconnectReason(Kernel::HLERequestContext& ctx) {
|
|
IPC::ResponseBuilder rb{ctx, 3};
|
|
rb.Push(ResultSuccess);
|
|
rb.PushEnum(lan_discovery.GetDisconnectReason());
|
|
}
|
|
|
|
void GetSecurityParameter(Kernel::HLERequestContext& ctx) {
|
|
SecurityParameter security_parameter{};
|
|
NetworkInfo info{};
|
|
const Result rc = lan_discovery.GetNetworkInfo(info);
|
|
|
|
if (rc.IsError()) {
|
|
LOG_ERROR(Service_LDN, "NetworkInfo is not valid {}", rc.raw);
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(rc);
|
|
return;
|
|
}
|
|
|
|
security_parameter.session_id = info.network_id.session_id;
|
|
std::memcpy(security_parameter.data.data(), info.ldn.security_parameter.data(),
|
|
sizeof(SecurityParameter::data));
|
|
|
|
IPC::ResponseBuilder rb{ctx, 10};
|
|
rb.Push(rc);
|
|
rb.PushRaw<SecurityParameter>(security_parameter);
|
|
}
|
|
|
|
void GetNetworkConfig(Kernel::HLERequestContext& ctx) {
|
|
NetworkConfig config{};
|
|
NetworkInfo info{};
|
|
const Result rc = lan_discovery.GetNetworkInfo(info);
|
|
|
|
if (rc.IsError()) {
|
|
LOG_ERROR(Service_LDN, "NetworkConfig is not valid {}", rc.raw);
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(rc);
|
|
return;
|
|
}
|
|
|
|
config.intent_id = info.network_id.intent_id;
|
|
config.channel = info.common.channel;
|
|
config.node_count_max = info.ldn.node_count_max;
|
|
config.local_communication_version = info.ldn.nodes[0].local_communication_version;
|
|
|
|
IPC::ResponseBuilder rb{ctx, 10};
|
|
rb.Push(rc);
|
|
rb.PushRaw<NetworkConfig>(config);
|
|
}
|
|
|
|
void AttachStateChangeEvent(Kernel::HLERequestContext& ctx) {
|
|
LOG_INFO(Service_LDN, "called");
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 1};
|
|
rb.Push(ResultSuccess);
|
|
rb.PushCopyObjects(state_change_event->GetReadableEvent());
|
|
}
|
|
|
|
void GetNetworkInfoLatestUpdate(Kernel::HLERequestContext& ctx) {
|
|
const std::size_t network_buffer_size = ctx.GetWriteBufferSize(0);
|
|
const std::size_t node_buffer_count = ctx.GetWriteBufferSize(1) / sizeof(NodeLatestUpdate);
|
|
|
|
if (node_buffer_count == 0 || network_buffer_size != sizeof(NetworkInfo)) {
|
|
LOG_ERROR(Service_LDN, "Invalid buffer, size = {}, count = {}", network_buffer_size,
|
|
node_buffer_count);
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(ResultBadInput);
|
|
return;
|
|
}
|
|
|
|
NetworkInfo info{};
|
|
std::vector<NodeLatestUpdate> latest_update(node_buffer_count);
|
|
|
|
const auto rc = lan_discovery.GetNetworkInfo(info, latest_update, latest_update.size());
|
|
if (rc.IsError()) {
|
|
LOG_ERROR(Service_LDN, "NetworkInfo is not valid {}", rc.raw);
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(rc);
|
|
return;
|
|
}
|
|
|
|
ctx.WriteBuffer(info, 0);
|
|
ctx.WriteBuffer(latest_update, 1);
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(ResultSuccess);
|
|
}
|
|
|
|
void Scan(Kernel::HLERequestContext& ctx) {
|
|
ScanImpl(ctx);
|
|
}
|
|
|
|
void ScanPrivate(Kernel::HLERequestContext& ctx) {
|
|
ScanImpl(ctx, true);
|
|
}
|
|
|
|
void ScanImpl(Kernel::HLERequestContext& ctx, bool is_private = false) {
|
|
IPC::RequestParser rp{ctx};
|
|
const auto channel{rp.PopEnum<WifiChannel>()};
|
|
const auto scan_filter{rp.PopRaw<ScanFilter>()};
|
|
|
|
const std::size_t network_info_size = ctx.GetWriteBufferSize() / sizeof(NetworkInfo);
|
|
|
|
if (network_info_size == 0) {
|
|
LOG_ERROR(Service_LDN, "Invalid buffer size {}", network_info_size);
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(ResultBadInput);
|
|
return;
|
|
}
|
|
|
|
u16 count = 0;
|
|
std::vector<NetworkInfo> network_infos(network_info_size);
|
|
Result rc = lan_discovery.Scan(network_infos, count, scan_filter);
|
|
|
|
LOG_INFO(Service_LDN,
|
|
"called, channel={}, filter_scan_flag={}, filter_network_type={}, is_private={}",
|
|
channel, scan_filter.flag, scan_filter.network_type, is_private);
|
|
|
|
ctx.WriteBuffer(network_infos);
|
|
|
|
IPC::ResponseBuilder rb{ctx, 3};
|
|
rb.Push(rc);
|
|
rb.Push<u32>(count);
|
|
}
|
|
|
|
void SetWirelessControllerRestriction(Kernel::HLERequestContext& ctx) {
|
|
LOG_WARNING(Service_LDN, "(STUBBED) called");
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(ResultSuccess);
|
|
}
|
|
|
|
void OpenAccessPoint(Kernel::HLERequestContext& ctx) {
|
|
LOG_INFO(Service_LDN, "called");
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(lan_discovery.OpenAccessPoint());
|
|
}
|
|
|
|
void CloseAccessPoint(Kernel::HLERequestContext& ctx) {
|
|
LOG_INFO(Service_LDN, "called");
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(lan_discovery.CloseAccessPoint());
|
|
}
|
|
|
|
void CreateNetwork(Kernel::HLERequestContext& ctx) {
|
|
LOG_INFO(Service_LDN, "called");
|
|
|
|
CreateNetworkImpl(ctx);
|
|
}
|
|
|
|
void CreateNetworkPrivate(Kernel::HLERequestContext& ctx) {
|
|
LOG_INFO(Service_LDN, "called");
|
|
|
|
CreateNetworkImpl(ctx, true);
|
|
}
|
|
|
|
void CreateNetworkImpl(Kernel::HLERequestContext& ctx, bool is_private = false) {
|
|
IPC::RequestParser rp{ctx};
|
|
|
|
const auto security_config{rp.PopRaw<SecurityConfig>()};
|
|
[[maybe_unused]] const auto security_parameter{is_private ? rp.PopRaw<SecurityParameter>()
|
|
: SecurityParameter{}};
|
|
const auto user_config{rp.PopRaw<UserConfig>()};
|
|
rp.Pop<u32>(); // Padding
|
|
const auto network_Config{rp.PopRaw<NetworkConfig>()};
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(lan_discovery.CreateNetwork(security_config, user_config, network_Config));
|
|
}
|
|
|
|
void DestroyNetwork(Kernel::HLERequestContext& ctx) {
|
|
LOG_INFO(Service_LDN, "called");
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(lan_discovery.DestroyNetwork());
|
|
}
|
|
|
|
void SetAdvertiseData(Kernel::HLERequestContext& ctx) {
|
|
std::vector<u8> read_buffer = ctx.ReadBuffer();
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(lan_discovery.SetAdvertiseData(read_buffer));
|
|
}
|
|
|
|
void SetStationAcceptPolicy(Kernel::HLERequestContext& ctx) {
|
|
LOG_WARNING(Service_LDN, "(STUBBED) called");
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(ResultSuccess);
|
|
}
|
|
|
|
void AddAcceptFilterEntry(Kernel::HLERequestContext& ctx) {
|
|
LOG_WARNING(Service_LDN, "(STUBBED) called");
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(ResultSuccess);
|
|
}
|
|
|
|
void OpenStation(Kernel::HLERequestContext& ctx) {
|
|
LOG_INFO(Service_LDN, "called");
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(lan_discovery.OpenStation());
|
|
}
|
|
|
|
void CloseStation(Kernel::HLERequestContext& ctx) {
|
|
LOG_INFO(Service_LDN, "called");
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(lan_discovery.CloseStation());
|
|
}
|
|
|
|
void Connect(Kernel::HLERequestContext& ctx) {
|
|
IPC::RequestParser rp{ctx};
|
|
struct Parameters {
|
|
SecurityConfig security_config;
|
|
UserConfig user_config;
|
|
u32 local_communication_version;
|
|
u32 option;
|
|
};
|
|
static_assert(sizeof(Parameters) == 0x7C, "Parameters has incorrect size.");
|
|
|
|
const auto parameters{rp.PopRaw<Parameters>()};
|
|
|
|
LOG_INFO(Service_LDN,
|
|
"called, passphrase_size={}, security_mode={}, "
|
|
"local_communication_version={}",
|
|
parameters.security_config.passphrase_size,
|
|
parameters.security_config.security_mode, parameters.local_communication_version);
|
|
|
|
const std::vector<u8> read_buffer = ctx.ReadBuffer();
|
|
if (read_buffer.size() != sizeof(NetworkInfo)) {
|
|
LOG_ERROR(Frontend, "NetworkInfo doesn't match read_buffer size!");
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(ResultBadInput);
|
|
return;
|
|
}
|
|
|
|
NetworkInfo network_info{};
|
|
std::memcpy(&network_info, read_buffer.data(), read_buffer.size());
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(lan_discovery.Connect(network_info, parameters.user_config,
|
|
static_cast<u16>(parameters.local_communication_version)));
|
|
}
|
|
|
|
void Disconnect(Kernel::HLERequestContext& ctx) {
|
|
LOG_INFO(Service_LDN, "called");
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(lan_discovery.Disconnect());
|
|
}
|
|
|
|
void Initialize(Kernel::HLERequestContext& ctx) {
|
|
const auto rc = InitializeImpl(ctx);
|
|
if (rc.IsError()) {
|
|
LOG_ERROR(Service_LDN, "Network isn't initialized, rc={}", rc.raw);
|
|
}
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(rc);
|
|
}
|
|
|
|
void Finalize(Kernel::HLERequestContext& ctx) {
|
|
if (auto room_member = room_network.GetRoomMember().lock()) {
|
|
room_member->Unbind(ldn_packet_received);
|
|
}
|
|
|
|
is_initialized = false;
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(lan_discovery.Finalize());
|
|
}
|
|
|
|
void Initialize2(Kernel::HLERequestContext& ctx) {
|
|
const auto rc = InitializeImpl(ctx);
|
|
if (rc.IsError()) {
|
|
LOG_ERROR(Service_LDN, "Network isn't initialized, rc={}", rc.raw);
|
|
}
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(rc);
|
|
}
|
|
|
|
Result InitializeImpl(Kernel::HLERequestContext& ctx) {
|
|
const auto network_interface = Network::GetSelectedNetworkInterface();
|
|
if (!network_interface) {
|
|
LOG_ERROR(Service_LDN, "No network interface is set");
|
|
return ResultAirplaneModeEnabled;
|
|
}
|
|
|
|
if (auto room_member = room_network.GetRoomMember().lock()) {
|
|
ldn_packet_received = room_member->BindOnLdnPacketReceived(
|
|
[this](const Network::LDNPacket& packet) { OnLDNPacketReceived(packet); });
|
|
} else {
|
|
LOG_ERROR(Service_LDN, "Couldn't bind callback!");
|
|
return ResultAirplaneModeEnabled;
|
|
}
|
|
|
|
lan_discovery.Initialize([&]() { OnEventFired(); });
|
|
is_initialized = true;
|
|
return ResultSuccess;
|
|
}
|
|
|
|
KernelHelpers::ServiceContext service_context;
|
|
Kernel::KEvent* state_change_event;
|
|
Network::RoomNetwork& room_network;
|
|
LANDiscovery lan_discovery;
|
|
|
|
// Callback identifier for the OnLDNPacketReceived event.
|
|
Network::RoomMember::CallbackHandle<Network::LDNPacket> ldn_packet_received;
|
|
|
|
bool is_initialized{};
|
|
};
|
|
|
|
class LDNS final : public ServiceFramework<LDNS> {
|
|
public:
|
|
explicit LDNS(Core::System& system_) : ServiceFramework{system_, "ldn:s"} {
|
|
// clang-format off
|
|
static const FunctionInfo functions[] = {
|
|
{0, &LDNS::CreateSystemLocalCommunicationService, "CreateSystemLocalCommunicationService"},
|
|
};
|
|
// clang-format on
|
|
|
|
RegisterHandlers(functions);
|
|
}
|
|
|
|
void CreateSystemLocalCommunicationService(Kernel::HLERequestContext& ctx) {
|
|
LOG_DEBUG(Service_LDN, "called");
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
|
rb.Push(ResultSuccess);
|
|
rb.PushIpcInterface<ISystemLocalCommunicationService>(system);
|
|
}
|
|
};
|
|
|
|
class LDNU final : public ServiceFramework<LDNU> {
|
|
public:
|
|
explicit LDNU(Core::System& system_) : ServiceFramework{system_, "ldn:u"} {
|
|
// clang-format off
|
|
static const FunctionInfo functions[] = {
|
|
{0, &LDNU::CreateUserLocalCommunicationService, "CreateUserLocalCommunicationService"},
|
|
};
|
|
// clang-format on
|
|
|
|
RegisterHandlers(functions);
|
|
}
|
|
|
|
void CreateUserLocalCommunicationService(Kernel::HLERequestContext& ctx) {
|
|
LOG_DEBUG(Service_LDN, "called");
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
|
rb.Push(ResultSuccess);
|
|
rb.PushIpcInterface<IUserLocalCommunicationService>(system);
|
|
}
|
|
};
|
|
|
|
class INetworkService final : public ServiceFramework<INetworkService> {
|
|
public:
|
|
explicit INetworkService(Core::System& system_) : ServiceFramework{system_, "INetworkService"} {
|
|
// clang-format off
|
|
static const FunctionInfo functions[] = {
|
|
{0, nullptr, "Initialize"},
|
|
{256, nullptr, "AttachNetworkInterfaceStateChangeEvent"},
|
|
{264, nullptr, "GetNetworkInterfaceLastError"},
|
|
{272, nullptr, "GetRole"},
|
|
{280, nullptr, "GetAdvertiseData"},
|
|
{288, nullptr, "GetGroupInfo"},
|
|
{296, nullptr, "GetGroupInfo2"},
|
|
{304, nullptr, "GetGroupOwner"},
|
|
{312, nullptr, "GetIpConfig"},
|
|
{320, nullptr, "GetLinkLevel"},
|
|
{512, nullptr, "Scan"},
|
|
{768, nullptr, "CreateGroup"},
|
|
{776, nullptr, "DestroyGroup"},
|
|
{784, nullptr, "SetAdvertiseData"},
|
|
{1536, nullptr, "SendToOtherGroup"},
|
|
{1544, nullptr, "RecvFromOtherGroup"},
|
|
{1552, nullptr, "AddAcceptableGroupId"},
|
|
{1560, nullptr, "ClearAcceptableGroupId"},
|
|
};
|
|
// clang-format on
|
|
|
|
RegisterHandlers(functions);
|
|
}
|
|
};
|
|
|
|
class INetworkServiceMonitor final : public ServiceFramework<INetworkServiceMonitor> {
|
|
public:
|
|
explicit INetworkServiceMonitor(Core::System& system_)
|
|
: ServiceFramework{system_, "INetworkServiceMonitor"} {
|
|
// clang-format off
|
|
static const FunctionInfo functions[] = {
|
|
{0, &INetworkServiceMonitor::Initialize, "Initialize"},
|
|
{256, nullptr, "AttachNetworkInterfaceStateChangeEvent"},
|
|
{264, nullptr, "GetNetworkInterfaceLastError"},
|
|
{272, nullptr, "GetRole"},
|
|
{280, nullptr, "GetAdvertiseData"},
|
|
{281, nullptr, "GetAdvertiseData2"},
|
|
{288, nullptr, "GetGroupInfo"},
|
|
{296, nullptr, "GetGroupInfo2"},
|
|
{304, nullptr, "GetGroupOwner"},
|
|
{312, nullptr, "GetIpConfig"},
|
|
{320, nullptr, "GetLinkLevel"},
|
|
{328, nullptr, "AttachJoinEvent"},
|
|
{336, nullptr, "GetMembers"},
|
|
};
|
|
// clang-format on
|
|
|
|
RegisterHandlers(functions);
|
|
}
|
|
|
|
void Initialize(Kernel::HLERequestContext& ctx) {
|
|
LOG_WARNING(Service_LDN, "(STUBBED) called");
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
rb.Push(ResultDisabled);
|
|
}
|
|
};
|
|
|
|
class LP2PAPP final : public ServiceFramework<LP2PAPP> {
|
|
public:
|
|
explicit LP2PAPP(Core::System& system_) : ServiceFramework{system_, "lp2p:app"} {
|
|
// clang-format off
|
|
static const FunctionInfo functions[] = {
|
|
{0, &LP2PAPP::CreateMonitorService, "CreateNetworkService"},
|
|
{8, &LP2PAPP::CreateMonitorService, "CreateNetworkServiceMonitor"},
|
|
};
|
|
// clang-format on
|
|
|
|
RegisterHandlers(functions);
|
|
}
|
|
|
|
void CreateNetworkervice(Kernel::HLERequestContext& ctx) {
|
|
IPC::RequestParser rp{ctx};
|
|
const u64 reserved_input = rp.Pop<u64>();
|
|
const u32 input = rp.Pop<u32>();
|
|
|
|
LOG_WARNING(Service_LDN, "(STUBBED) called reserved_input={} input={}", reserved_input,
|
|
input);
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
|
rb.Push(ResultSuccess);
|
|
rb.PushIpcInterface<INetworkService>(system);
|
|
}
|
|
|
|
void CreateMonitorService(Kernel::HLERequestContext& ctx) {
|
|
IPC::RequestParser rp{ctx};
|
|
const u64 reserved_input = rp.Pop<u64>();
|
|
|
|
LOG_WARNING(Service_LDN, "(STUBBED) called reserved_input={}", reserved_input);
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
|
rb.Push(ResultSuccess);
|
|
rb.PushIpcInterface<INetworkServiceMonitor>(system);
|
|
}
|
|
};
|
|
|
|
class LP2PSYS final : public ServiceFramework<LP2PSYS> {
|
|
public:
|
|
explicit LP2PSYS(Core::System& system_) : ServiceFramework{system_, "lp2p:sys"} {
|
|
// clang-format off
|
|
static const FunctionInfo functions[] = {
|
|
{0, &LP2PSYS::CreateMonitorService, "CreateNetworkService"},
|
|
{8, &LP2PSYS::CreateMonitorService, "CreateNetworkServiceMonitor"},
|
|
};
|
|
// clang-format on
|
|
|
|
RegisterHandlers(functions);
|
|
}
|
|
|
|
void CreateNetworkervice(Kernel::HLERequestContext& ctx) {
|
|
IPC::RequestParser rp{ctx};
|
|
const u64 reserved_input = rp.Pop<u64>();
|
|
const u32 input = rp.Pop<u32>();
|
|
|
|
LOG_WARNING(Service_LDN, "(STUBBED) called reserved_input={} input={}", reserved_input,
|
|
input);
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
|
rb.Push(ResultSuccess);
|
|
rb.PushIpcInterface<INetworkService>(system);
|
|
}
|
|
|
|
void CreateMonitorService(Kernel::HLERequestContext& ctx) {
|
|
IPC::RequestParser rp{ctx};
|
|
const u64 reserved_input = rp.Pop<u64>();
|
|
|
|
LOG_WARNING(Service_LDN, "(STUBBED) called reserved_input={}", reserved_input);
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
|
rb.Push(ResultSuccess);
|
|
rb.PushIpcInterface<INetworkServiceMonitor>(system);
|
|
}
|
|
};
|
|
|
|
void InstallInterfaces(SM::ServiceManager& sm, Core::System& system) {
|
|
std::make_shared<LDNM>(system)->InstallAsService(sm);
|
|
std::make_shared<LDNS>(system)->InstallAsService(sm);
|
|
std::make_shared<LDNU>(system)->InstallAsService(sm);
|
|
std::make_shared<LP2PAPP>(system)->InstallAsService(sm);
|
|
std::make_shared<LP2PSYS>(system)->InstallAsService(sm);
|
|
}
|
|
|
|
} // namespace Service::LDN
|