forked from eden-emu/eden
		
	ldn: Initial implementation
This commit is contained in:
		
							parent
							
								
									c48259ca81
								
							
						
					
					
						commit
						48d6226115
					
				
					 15 changed files with 1134 additions and 126 deletions
				
			
		|  | @ -500,6 +500,8 @@ add_library(core STATIC | ||||||
|     hle/service/jit/jit.h |     hle/service/jit/jit.h | ||||||
|     hle/service/lbl/lbl.cpp |     hle/service/lbl/lbl.cpp | ||||||
|     hle/service/lbl/lbl.h |     hle/service/lbl/lbl.h | ||||||
|  |     hle/service/ldn/lan_discovery.cpp | ||||||
|  |     hle/service/ldn/lan_discovery.h | ||||||
|     hle/service/ldn/ldn_results.h |     hle/service/ldn/ldn_results.h | ||||||
|     hle/service/ldn/ldn.cpp |     hle/service/ldn/ldn.cpp | ||||||
|     hle/service/ldn/ldn.h |     hle/service/ldn/ldn.h | ||||||
|  |  | ||||||
							
								
								
									
										644
									
								
								src/core/hle/service/ldn/lan_discovery.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										644
									
								
								src/core/hle/service/ldn/lan_discovery.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,644 @@ | ||||||
|  | // SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
 | ||||||
|  | // SPDX-License-Identifier: GPL-2.0-or-later
 | ||||||
|  | 
 | ||||||
|  | #include "core/hle/service/ldn/lan_discovery.h" | ||||||
|  | #include "core/internal_network/network.h" | ||||||
|  | #include "core/internal_network/network_interface.h" | ||||||
|  | 
 | ||||||
|  | namespace Service::LDN { | ||||||
|  | 
 | ||||||
|  | LanStation::LanStation(s8 node_id_, LANDiscovery* discovery_) | ||||||
|  |     : node_info(nullptr), status(NodeStatus::Disconnected), node_id(node_id_), | ||||||
|  |       discovery(discovery_) {} | ||||||
|  | 
 | ||||||
|  | LanStation::~LanStation() = default; | ||||||
|  | 
 | ||||||
|  | NodeStatus LanStation::GetStatus() const { | ||||||
|  |     return status; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LanStation::OnClose() { | ||||||
|  |     LOG_INFO(Service_LDN, "OnClose {}", node_id); | ||||||
|  |     Reset(); | ||||||
|  |     discovery->UpdateNodes(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LanStation::Reset() { | ||||||
|  |     status = NodeStatus::Disconnected; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | void LanStation::OverrideInfo() { | ||||||
|  |     bool connected = GetStatus() == NodeStatus::Connected; | ||||||
|  |     node_info->node_id = node_id; | ||||||
|  |     node_info->is_connected = connected ? 1 : 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | LANDiscovery::LANDiscovery(Network::RoomNetwork& room_network_) | ||||||
|  |     : stations({{{1, this}, {2, this}, {3, this}, {4, this}, {5, this}, {6, this}, {7, this}}}), | ||||||
|  |       room_network{room_network_} { | ||||||
|  |     LOG_INFO(Service_LDN, "LANDiscovery"); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | LANDiscovery::~LANDiscovery() { | ||||||
|  |     LOG_INFO(Service_LDN, "~LANDiscovery"); | ||||||
|  |     if (inited) { | ||||||
|  |         Result rc = Finalize(); | ||||||
|  |         LOG_INFO(Service_LDN, "Finalize: {}", rc.raw); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LANDiscovery::InitNetworkInfo() { | ||||||
|  |     network_info.common.bssid = GetFakeMac(); | ||||||
|  |     network_info.common.channel = WifiChannel::Wifi24_6; | ||||||
|  |     network_info.common.link_level = LinkLevel::Good; | ||||||
|  |     network_info.common.network_type = PackedNetworkType::Ldn; | ||||||
|  |     network_info.common.ssid = fake_ssid; | ||||||
|  | 
 | ||||||
|  |     auto& nodes = network_info.ldn.nodes; | ||||||
|  |     for (std::size_t i = 0; i < NodeCountMax; i++) { | ||||||
|  |         nodes[i].node_id = static_cast<s8>(i); | ||||||
|  |         nodes[i].is_connected = 0; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LANDiscovery::InitNodeStateChange() { | ||||||
|  |     for (auto& node_update : nodeChanges) { | ||||||
|  |         node_update.state_change = NodeStateChange::None; | ||||||
|  |     } | ||||||
|  |     for (auto& node_state : node_last_states) { | ||||||
|  |         node_state = 0; | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | State LANDiscovery::GetState() const { | ||||||
|  |     return state; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LANDiscovery::SetState(State new_state) { | ||||||
|  |     state = new_state; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result LANDiscovery::GetNetworkInfo(NetworkInfo& out_network) const { | ||||||
|  |     if (state == State::AccessPointCreated || state == State::StationConnected) { | ||||||
|  |         std::memcpy(&out_network, &network_info, sizeof(network_info)); | ||||||
|  |         return ResultSuccess; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return ResultBadState; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result LANDiscovery::GetNetworkInfo(NetworkInfo& out_network, | ||||||
|  |                                     std::vector<NodeLatestUpdate>& out_updates, | ||||||
|  |                                     std::size_t buffer_count) { | ||||||
|  |     if (buffer_count > NodeCountMax) { | ||||||
|  |         return ResultInvalidBufferCount; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (state == State::AccessPointCreated || state == State::StationConnected) { | ||||||
|  |         std::memcpy(&out_network, &network_info, sizeof(network_info)); | ||||||
|  |         for (std::size_t i = 0; i < buffer_count; i++) { | ||||||
|  |             out_updates[i].state_change = nodeChanges[i].state_change; | ||||||
|  |             nodeChanges[i].state_change = NodeStateChange::None; | ||||||
|  |         } | ||||||
|  |         return ResultSuccess; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return ResultBadState; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | DisconnectReason LANDiscovery::GetDisconnectReason() const { | ||||||
|  |     return disconnect_reason; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result LANDiscovery::Scan(std::vector<NetworkInfo>& networks, u16& count, | ||||||
|  |                           const ScanFilter& filter) { | ||||||
|  |     if (!IsFlagSet(filter.flag, ScanFilterFlag::NetworkType) || | ||||||
|  |         filter.network_type <= NetworkType::All) { | ||||||
|  |         if (!IsFlagSet(filter.flag, ScanFilterFlag::Ssid) && filter.ssid.length >= SsidLengthMax) { | ||||||
|  |             return ResultBadInput; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     { | ||||||
|  |         std::scoped_lock lock{packet_mutex}; | ||||||
|  |         scan_results.clear(); | ||||||
|  | 
 | ||||||
|  |         SendBroadcast(Network::LDNPacketType::Scan); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     LOG_INFO(Service_LDN, "Waiting for scan replies"); | ||||||
|  |     std::this_thread::sleep_for(std::chrono::seconds(1)); | ||||||
|  | 
 | ||||||
|  |     std::scoped_lock lock{packet_mutex}; | ||||||
|  |     for (const auto& [key, info] : scan_results) { | ||||||
|  |         if (count >= networks.size()) { | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (IsFlagSet(filter.flag, ScanFilterFlag::LocalCommunicationId)) { | ||||||
|  |             if (filter.network_id.intent_id.local_communication_id != | ||||||
|  |                 info.network_id.intent_id.local_communication_id) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         if (IsFlagSet(filter.flag, ScanFilterFlag::SessionId)) { | ||||||
|  |             if (filter.network_id.session_id != info.network_id.session_id) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         if (IsFlagSet(filter.flag, ScanFilterFlag::NetworkType)) { | ||||||
|  |             if (filter.network_type != static_cast<NetworkType>(info.common.network_type)) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         if (IsFlagSet(filter.flag, ScanFilterFlag::Ssid)) { | ||||||
|  |             if (filter.ssid != info.common.ssid) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         if (IsFlagSet(filter.flag, ScanFilterFlag::SceneId)) { | ||||||
|  |             if (filter.network_id.intent_id.scene_id != info.network_id.intent_id.scene_id) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         networks[count++] = info; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return ResultSuccess; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result LANDiscovery::SetAdvertiseData(std::vector<u8>& data) { | ||||||
|  |     std::scoped_lock lock{packet_mutex}; | ||||||
|  |     std::size_t size = data.size(); | ||||||
|  |     if (size > AdvertiseDataSizeMax) { | ||||||
|  |         return ResultAdvertiseDataTooLarge; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     std::memcpy(network_info.ldn.advertise_data.data(), data.data(), size); | ||||||
|  |     network_info.ldn.advertise_data_size = static_cast<u16>(size); | ||||||
|  | 
 | ||||||
|  |     UpdateNodes(); | ||||||
|  | 
 | ||||||
|  |     return ResultSuccess; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result LANDiscovery::OpenAccessPoint() { | ||||||
|  |     std::scoped_lock lock{packet_mutex}; | ||||||
|  |     disconnect_reason = DisconnectReason::None; | ||||||
|  |     if (state == State::None) { | ||||||
|  |         return ResultBadState; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ResetStations(); | ||||||
|  |     SetState(State::AccessPointOpened); | ||||||
|  | 
 | ||||||
|  |     return ResultSuccess; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result LANDiscovery::CloseAccessPoint() { | ||||||
|  |     std::scoped_lock lock{packet_mutex}; | ||||||
|  |     if (state == State::None) { | ||||||
|  |         return ResultBadState; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (state == State::AccessPointCreated) { | ||||||
|  |         DestroyNetwork(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ResetStations(); | ||||||
|  |     SetState(State::Initialized); | ||||||
|  | 
 | ||||||
|  |     return ResultSuccess; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result LANDiscovery::OpenStation() { | ||||||
|  |     std::scoped_lock lock{packet_mutex}; | ||||||
|  |     disconnect_reason = DisconnectReason::None; | ||||||
|  |     if (state == State::None) { | ||||||
|  |         return ResultBadState; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ResetStations(); | ||||||
|  |     SetState(State::StationOpened); | ||||||
|  | 
 | ||||||
|  |     return ResultSuccess; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result LANDiscovery::CloseStation() { | ||||||
|  |     std::scoped_lock lock{packet_mutex}; | ||||||
|  |     if (state == State::None) { | ||||||
|  |         return ResultBadState; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (state == State::StationConnected) { | ||||||
|  |         Disconnect(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ResetStations(); | ||||||
|  |     SetState(State::Initialized); | ||||||
|  | 
 | ||||||
|  |     return ResultSuccess; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result LANDiscovery::CreateNetwork(const SecurityConfig& security_config, | ||||||
|  |                                    const UserConfig& user_config, | ||||||
|  |                                    const NetworkConfig& network_config) { | ||||||
|  |     std::scoped_lock lock{packet_mutex}; | ||||||
|  | 
 | ||||||
|  |     if (state != State::AccessPointOpened) { | ||||||
|  |         return ResultBadState; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     InitNetworkInfo(); | ||||||
|  |     network_info.ldn.node_count_max = network_config.node_count_max; | ||||||
|  |     network_info.ldn.security_mode = security_config.security_mode; | ||||||
|  | 
 | ||||||
|  |     if (network_config.channel == WifiChannel::Default) { | ||||||
|  |         network_info.common.channel = WifiChannel::Wifi24_6; | ||||||
|  |     } else { | ||||||
|  |         network_info.common.channel = network_config.channel; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     std::independent_bits_engine<std::mt19937, 64, u64> bits_engine; | ||||||
|  |     network_info.network_id.session_id.high = bits_engine(); | ||||||
|  |     network_info.network_id.session_id.low = bits_engine(); | ||||||
|  |     network_info.network_id.intent_id = network_config.intent_id; | ||||||
|  | 
 | ||||||
|  |     NodeInfo& node0 = network_info.ldn.nodes[0]; | ||||||
|  |     const Result rc2 = GetNodeInfo(node0, user_config, network_config.local_communication_version); | ||||||
|  |     if (rc2.IsError()) { | ||||||
|  |         return ResultAccessPointConnectionFailed; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     SetState(State::AccessPointCreated); | ||||||
|  | 
 | ||||||
|  |     InitNodeStateChange(); | ||||||
|  |     node0.is_connected = 1; | ||||||
|  |     UpdateNodes(); | ||||||
|  | 
 | ||||||
|  |     return rc2; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result LANDiscovery::DestroyNetwork() { | ||||||
|  |     for (auto local_ip : connected_clients) { | ||||||
|  |         SendPacket(Network::LDNPacketType::DestroyNetwork, local_ip); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     ResetStations(); | ||||||
|  | 
 | ||||||
|  |     SetState(State::AccessPointOpened); | ||||||
|  |     LanEvent(); | ||||||
|  | 
 | ||||||
|  |     return ResultSuccess; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result LANDiscovery::Connect(const NetworkInfo& network_info_, const UserConfig& user_config, | ||||||
|  |                              u16 local_communication_version) { | ||||||
|  |     std::scoped_lock lock{packet_mutex}; | ||||||
|  |     if (network_info_.ldn.node_count == 0) { | ||||||
|  |         return ResultInvalidNodeCount; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     Result rc = GetNodeInfo(node_info, user_config, local_communication_version); | ||||||
|  |     if (rc.IsError()) { | ||||||
|  |         return ResultConnectionFailed; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     Ipv4Address node_host = network_info_.ldn.nodes[0].ipv4_address; | ||||||
|  |     std::reverse(std::begin(node_host), std::end(node_host)); // htonl
 | ||||||
|  |     host_ip = node_host; | ||||||
|  |     SendPacket(Network::LDNPacketType::Connect, node_info, *host_ip); | ||||||
|  | 
 | ||||||
|  |     InitNodeStateChange(); | ||||||
|  | 
 | ||||||
|  |     std::this_thread::sleep_for(std::chrono::seconds(1)); | ||||||
|  | 
 | ||||||
|  |     return ResultSuccess; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result LANDiscovery::Disconnect() { | ||||||
|  |     if (host_ip) { | ||||||
|  |         SendPacket(Network::LDNPacketType::Disconnect, node_info, *host_ip); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     SetState(State::StationOpened); | ||||||
|  |     LanEvent(); | ||||||
|  | 
 | ||||||
|  |     return ResultSuccess; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result LANDiscovery::Initialize(LanEventFunc lan_event, bool listening) { | ||||||
|  |     std::scoped_lock lock{packet_mutex}; | ||||||
|  |     if (inited) { | ||||||
|  |         return ResultSuccess; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     for (auto& station : stations) { | ||||||
|  |         station.discovery = this; | ||||||
|  |         station.node_info = &network_info.ldn.nodes[station.node_id]; | ||||||
|  |         station.Reset(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     connected_clients.clear(); | ||||||
|  |     LanEvent = lan_event; | ||||||
|  | 
 | ||||||
|  |     SetState(State::Initialized); | ||||||
|  | 
 | ||||||
|  |     inited = true; | ||||||
|  |     return ResultSuccess; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result LANDiscovery::Finalize() { | ||||||
|  |     std::scoped_lock lock{packet_mutex}; | ||||||
|  |     Result rc = ResultSuccess; | ||||||
|  | 
 | ||||||
|  |     if (inited) { | ||||||
|  |         if (state == State::AccessPointCreated) { | ||||||
|  |             DestroyNetwork(); | ||||||
|  |         } | ||||||
|  |         if (state == State::StationConnected) { | ||||||
|  |             Disconnect(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         ResetStations(); | ||||||
|  |         inited = false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     SetState(State::None); | ||||||
|  | 
 | ||||||
|  |     return rc; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LANDiscovery::ResetStations() { | ||||||
|  |     for (auto& station : stations) { | ||||||
|  |         station.Reset(); | ||||||
|  |     } | ||||||
|  |     connected_clients.clear(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LANDiscovery::UpdateNodes() { | ||||||
|  |     u8 count = 0; | ||||||
|  |     for (auto& station : stations) { | ||||||
|  |         bool connected = station.GetStatus() == NodeStatus::Connected; | ||||||
|  |         if (connected) { | ||||||
|  |             count++; | ||||||
|  |         } | ||||||
|  |         station.OverrideInfo(); | ||||||
|  |     } | ||||||
|  |     network_info.ldn.node_count = count + 1; | ||||||
|  | 
 | ||||||
|  |     for (auto local_ip : connected_clients) { | ||||||
|  |         SendPacket(Network::LDNPacketType::SyncNetwork, network_info, local_ip); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     OnNetworkInfoChanged(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LANDiscovery::OnSyncNetwork(const NetworkInfo& info) { | ||||||
|  |     network_info = info; | ||||||
|  |     if (state == State::StationOpened) { | ||||||
|  |         SetState(State::StationConnected); | ||||||
|  |     } | ||||||
|  |     OnNetworkInfoChanged(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LANDiscovery::OnDisconnectFromHost() { | ||||||
|  |     LOG_INFO(Service_LDN, "OnDisconnectFromHost state: {}", static_cast<int>(state)); | ||||||
|  |     host_ip = std::nullopt; | ||||||
|  |     if (state == State::StationConnected) { | ||||||
|  |         SetState(State::StationOpened); | ||||||
|  |         LanEvent(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LANDiscovery::OnNetworkInfoChanged() { | ||||||
|  |     if (IsNodeStateChanged()) { | ||||||
|  |         LanEvent(); | ||||||
|  |     } | ||||||
|  |     return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Network::IPv4Address LANDiscovery::GetLocalIp() const { | ||||||
|  |     Network::IPv4Address local_ip{0xFF, 0xFF, 0xFF, 0xFF}; | ||||||
|  |     if (auto room_member = room_network.GetRoomMember().lock()) { | ||||||
|  |         if (room_member->IsConnected()) { | ||||||
|  |             local_ip = room_member->GetFakeIpAddress(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     return local_ip; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | template <typename Data> | ||||||
|  | void LANDiscovery::SendPacket(Network::LDNPacketType type, const Data& data, | ||||||
|  |                               Ipv4Address remote_ip) { | ||||||
|  |     Network::LDNPacket packet; | ||||||
|  |     packet.type = type; | ||||||
|  | 
 | ||||||
|  |     packet.broadcast = false; | ||||||
|  |     packet.local_ip = GetLocalIp(); | ||||||
|  |     packet.remote_ip = remote_ip; | ||||||
|  | 
 | ||||||
|  |     packet.data.clear(); | ||||||
|  |     packet.data.resize(sizeof(data)); | ||||||
|  |     std::memcpy(packet.data.data(), &data, sizeof(data)); | ||||||
|  |     SendPacket(packet); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LANDiscovery::SendPacket(Network::LDNPacketType type, Ipv4Address remote_ip) { | ||||||
|  |     Network::LDNPacket packet; | ||||||
|  |     packet.type = type; | ||||||
|  | 
 | ||||||
|  |     packet.broadcast = false; | ||||||
|  |     packet.local_ip = GetLocalIp(); | ||||||
|  |     packet.remote_ip = remote_ip; | ||||||
|  | 
 | ||||||
|  |     packet.data.clear(); | ||||||
|  |     SendPacket(packet); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | template <typename Data> | ||||||
|  | void LANDiscovery::SendBroadcast(Network::LDNPacketType type, const Data& data) { | ||||||
|  |     Network::LDNPacket packet; | ||||||
|  |     packet.type = type; | ||||||
|  | 
 | ||||||
|  |     packet.broadcast = true; | ||||||
|  |     packet.local_ip = GetLocalIp(); | ||||||
|  | 
 | ||||||
|  |     packet.data.clear(); | ||||||
|  |     packet.data.resize(sizeof(data)); | ||||||
|  |     std::memcpy(packet.data.data(), &data, sizeof(data)); | ||||||
|  |     SendPacket(packet); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LANDiscovery::SendBroadcast(Network::LDNPacketType type) { | ||||||
|  |     Network::LDNPacket packet; | ||||||
|  |     packet.type = type; | ||||||
|  | 
 | ||||||
|  |     packet.broadcast = true; | ||||||
|  |     packet.local_ip = GetLocalIp(); | ||||||
|  | 
 | ||||||
|  |     packet.data.clear(); | ||||||
|  |     SendPacket(packet); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LANDiscovery::SendPacket(const Network::LDNPacket& packet) { | ||||||
|  |     if (auto room_member = room_network.GetRoomMember().lock()) { | ||||||
|  |         if (room_member->IsConnected()) { | ||||||
|  |             room_member->SendLdnPacket(packet); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void LANDiscovery::ReceivePacket(const Network::LDNPacket& packet) { | ||||||
|  |     std::scoped_lock lock{packet_mutex}; | ||||||
|  |     switch (packet.type) { | ||||||
|  |     case Network::LDNPacketType::Scan: { | ||||||
|  |         LOG_INFO(Frontend, "Scan packet received!"); | ||||||
|  |         if (state == State::AccessPointCreated) { | ||||||
|  |             // Reply to the sender
 | ||||||
|  |             SendPacket(Network::LDNPacketType::ScanResp, network_info, packet.local_ip); | ||||||
|  |         } | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |     case Network::LDNPacketType::ScanResp: { | ||||||
|  |         LOG_INFO(Frontend, "ScanResp packet received!"); | ||||||
|  | 
 | ||||||
|  |         NetworkInfo info{}; | ||||||
|  |         std::memcpy(&info, packet.data.data(), sizeof(NetworkInfo)); | ||||||
|  |         scan_results.insert({info.common.bssid, info}); | ||||||
|  | 
 | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |     case Network::LDNPacketType::Connect: { | ||||||
|  |         LOG_INFO(Frontend, "Connect packet received!"); | ||||||
|  | 
 | ||||||
|  |         NodeInfo info{}; | ||||||
|  |         std::memcpy(&info, packet.data.data(), sizeof(NodeInfo)); | ||||||
|  | 
 | ||||||
|  |         connected_clients.push_back(packet.local_ip); | ||||||
|  | 
 | ||||||
|  |         for (LanStation& station : stations) { | ||||||
|  |             if (station.status != NodeStatus::Connected) { | ||||||
|  |                 *station.node_info = info; | ||||||
|  |                 station.status = NodeStatus::Connected; | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         UpdateNodes(); | ||||||
|  | 
 | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |     case Network::LDNPacketType::Disconnect: { | ||||||
|  |         LOG_INFO(Frontend, "Disconnect packet received!"); | ||||||
|  | 
 | ||||||
|  |         connected_clients.erase( | ||||||
|  |             std::remove(connected_clients.begin(), connected_clients.end(), packet.local_ip), | ||||||
|  |             connected_clients.end()); | ||||||
|  | 
 | ||||||
|  |         NodeInfo info{}; | ||||||
|  |         std::memcpy(&info, packet.data.data(), sizeof(NodeInfo)); | ||||||
|  | 
 | ||||||
|  |         for (LanStation& station : stations) { | ||||||
|  |             if (station.status == NodeStatus::Connected && | ||||||
|  |                 station.node_info->mac_address == info.mac_address) { | ||||||
|  |                 station.OnClose(); | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |     case Network::LDNPacketType::DestroyNetwork: { | ||||||
|  |         ResetStations(); | ||||||
|  |         OnDisconnectFromHost(); | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |     case Network::LDNPacketType::SyncNetwork: { | ||||||
|  |         if (state == State::StationOpened || state == State::StationConnected) { | ||||||
|  |             LOG_INFO(Frontend, "SyncNetwork packet received!"); | ||||||
|  |             NetworkInfo info{}; | ||||||
|  |             std::memcpy(&info, packet.data.data(), sizeof(NetworkInfo)); | ||||||
|  | 
 | ||||||
|  |             OnSyncNetwork(info); | ||||||
|  |         } else { | ||||||
|  |             LOG_INFO(Frontend, "SyncNetwork packet received but in wrong State!"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |     default: { | ||||||
|  |         LOG_INFO(Frontend, "ReceivePacket unhandled type {}", static_cast<int>(packet.type)); | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool LANDiscovery::IsNodeStateChanged() { | ||||||
|  |     bool changed = false; | ||||||
|  |     const auto& nodes = network_info.ldn.nodes; | ||||||
|  |     for (int i = 0; i < NodeCountMax; i++) { | ||||||
|  |         if (nodes[i].is_connected != node_last_states[i]) { | ||||||
|  |             if (nodes[i].is_connected) { | ||||||
|  |                 nodeChanges[i].state_change |= NodeStateChange::Connect; | ||||||
|  |             } else { | ||||||
|  |                 nodeChanges[i].state_change |= NodeStateChange::Disconnect; | ||||||
|  |             } | ||||||
|  |             node_last_states[i] = nodes[i].is_connected; | ||||||
|  |             changed = true; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     return changed; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool LANDiscovery::IsFlagSet(ScanFilterFlag flag, ScanFilterFlag search_flag) const { | ||||||
|  |     const auto flag_value = static_cast<u32>(flag); | ||||||
|  |     const auto search_flag_value = static_cast<u32>(search_flag); | ||||||
|  |     return (flag_value & search_flag_value) == search_flag_value; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int LANDiscovery::GetStationCount() { | ||||||
|  |     int count = 0; | ||||||
|  |     for (const auto& station : stations) { | ||||||
|  |         if (station.GetStatus() != NodeStatus::Disconnected) { | ||||||
|  |             count++; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return count; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | MacAddress LANDiscovery::GetFakeMac() const { | ||||||
|  |     MacAddress mac{}; | ||||||
|  |     mac.raw[0] = 0x02; | ||||||
|  |     mac.raw[1] = 0x00; | ||||||
|  | 
 | ||||||
|  |     const auto ip = GetLocalIp(); | ||||||
|  |     memcpy(mac.raw.data() + 2, &ip, sizeof(ip)); | ||||||
|  | 
 | ||||||
|  |     return mac; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Result LANDiscovery::GetNodeInfo(NodeInfo& node, const UserConfig& userConfig, | ||||||
|  |                                  u16 localCommunicationVersion) { | ||||||
|  |     const auto network_interface = Network::GetSelectedNetworkInterface(); | ||||||
|  | 
 | ||||||
|  |     if (!network_interface) { | ||||||
|  |         LOG_ERROR(Service_LDN, "No network interface available"); | ||||||
|  |         return ResultNoIpAddress; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     node.mac_address = GetFakeMac(); | ||||||
|  |     node.is_connected = 1; | ||||||
|  |     std::memcpy(node.user_name.data(), userConfig.user_name.data(), UserNameBytesMax + 1); | ||||||
|  |     node.local_communication_version = localCommunicationVersion; | ||||||
|  | 
 | ||||||
|  |     Ipv4Address current_address = GetLocalIp(); | ||||||
|  |     std::reverse(std::begin(current_address), std::end(current_address)); // ntohl
 | ||||||
|  |     node.ipv4_address = current_address; | ||||||
|  | 
 | ||||||
|  |     return ResultSuccess; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | } // namespace Service::LDN
 | ||||||
							
								
								
									
										133
									
								
								src/core/hle/service/ldn/lan_discovery.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										133
									
								
								src/core/hle/service/ldn/lan_discovery.h
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,133 @@ | ||||||
|  | // SPDX-FileCopyrightText: Copyright 2022 yuzu Emulator Project
 | ||||||
|  | // SPDX-License-Identifier: GPL-2.0-or-later
 | ||||||
|  | 
 | ||||||
|  | #pragma once | ||||||
|  | 
 | ||||||
|  | #include <array> | ||||||
|  | #include <cstring> | ||||||
|  | #include <functional> | ||||||
|  | #include <memory> | ||||||
|  | #include <mutex> | ||||||
|  | #include <optional> | ||||||
|  | #include <random> | ||||||
|  | #include <thread> | ||||||
|  | #include <unordered_map> | ||||||
|  | 
 | ||||||
|  | #include "common/logging/log.h" | ||||||
|  | #include "common/socket_types.h" | ||||||
|  | #include "core/hle/result.h" | ||||||
|  | #include "core/hle/service/ldn/ldn_results.h" | ||||||
|  | #include "core/hle/service/ldn/ldn_types.h" | ||||||
|  | #include "network/network.h" | ||||||
|  | 
 | ||||||
|  | namespace Service::LDN { | ||||||
|  | 
 | ||||||
|  | class LANDiscovery; | ||||||
|  | 
 | ||||||
|  | class LanStation { | ||||||
|  | public: | ||||||
|  |     LanStation(s8 node_id_, LANDiscovery* discovery_); | ||||||
|  |     ~LanStation(); | ||||||
|  | 
 | ||||||
|  |     void OnClose(); | ||||||
|  |     NodeStatus GetStatus() const; | ||||||
|  |     void Reset(); | ||||||
|  |     void OverrideInfo(); | ||||||
|  | 
 | ||||||
|  | protected: | ||||||
|  |     friend class LANDiscovery; | ||||||
|  |     NodeInfo* node_info; | ||||||
|  |     NodeStatus status; | ||||||
|  |     s8 node_id; | ||||||
|  |     LANDiscovery* discovery; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | class LANDiscovery { | ||||||
|  | public: | ||||||
|  |     typedef std::function<void()> LanEventFunc; | ||||||
|  | 
 | ||||||
|  |     LANDiscovery(Network::RoomNetwork& room_network_); | ||||||
|  |     ~LANDiscovery(); | ||||||
|  | 
 | ||||||
|  |     State GetState() const; | ||||||
|  |     void SetState(State new_state); | ||||||
|  | 
 | ||||||
|  |     Result GetNetworkInfo(NetworkInfo& out_network) const; | ||||||
|  |     Result GetNetworkInfo(NetworkInfo& out_network, std::vector<NodeLatestUpdate>& out_updates, | ||||||
|  |                           std::size_t buffer_count); | ||||||
|  | 
 | ||||||
|  |     DisconnectReason GetDisconnectReason() const; | ||||||
|  |     Result Scan(std::vector<NetworkInfo>& networks, u16& count, const ScanFilter& filter); | ||||||
|  |     Result SetAdvertiseData(std::vector<u8>& data); | ||||||
|  | 
 | ||||||
|  |     Result OpenAccessPoint(); | ||||||
|  |     Result CloseAccessPoint(); | ||||||
|  | 
 | ||||||
|  |     Result OpenStation(); | ||||||
|  |     Result CloseStation(); | ||||||
|  | 
 | ||||||
|  |     Result CreateNetwork(const SecurityConfig& security_config, const UserConfig& user_config, | ||||||
|  |                          const NetworkConfig& network_config); | ||||||
|  |     Result DestroyNetwork(); | ||||||
|  | 
 | ||||||
|  |     Result Connect(const NetworkInfo& network_info_, const UserConfig& user_config, | ||||||
|  |                    u16 local_communication_version); | ||||||
|  |     Result Disconnect(); | ||||||
|  | 
 | ||||||
|  |     Result Initialize(LanEventFunc lan_event = empty_func, bool listening = true); | ||||||
|  |     Result Finalize(); | ||||||
|  | 
 | ||||||
|  |     void ReceivePacket(const Network::LDNPacket& packet); | ||||||
|  | 
 | ||||||
|  | protected: | ||||||
|  |     friend class LanStation; | ||||||
|  | 
 | ||||||
|  |     void InitNetworkInfo(); | ||||||
|  |     void InitNodeStateChange(); | ||||||
|  | 
 | ||||||
|  |     void ResetStations(); | ||||||
|  |     void UpdateNodes(); | ||||||
|  | 
 | ||||||
|  |     void OnSyncNetwork(const NetworkInfo& info); | ||||||
|  |     void OnDisconnectFromHost(); | ||||||
|  |     void OnNetworkInfoChanged(); | ||||||
|  | 
 | ||||||
|  |     bool IsNodeStateChanged(); | ||||||
|  |     bool IsFlagSet(ScanFilterFlag flag, ScanFilterFlag search_flag) const; | ||||||
|  |     int GetStationCount(); | ||||||
|  |     MacAddress GetFakeMac() const; | ||||||
|  |     Result GetNodeInfo(NodeInfo& node, const UserConfig& user_config, | ||||||
|  |                        u16 local_communication_version); | ||||||
|  | 
 | ||||||
|  |     Network::IPv4Address GetLocalIp() const; | ||||||
|  |     template <typename Data> | ||||||
|  |     void SendPacket(Network::LDNPacketType type, const Data& data, Ipv4Address remote_ip); | ||||||
|  |     void SendPacket(Network::LDNPacketType type, Ipv4Address remote_ip); | ||||||
|  |     template <typename Data> | ||||||
|  |     void SendBroadcast(Network::LDNPacketType type, const Data& data); | ||||||
|  |     void SendBroadcast(Network::LDNPacketType type); | ||||||
|  |     void SendPacket(const Network::LDNPacket& packet); | ||||||
|  | 
 | ||||||
|  |     static const LanEventFunc empty_func; | ||||||
|  |     const Ssid fake_ssid{"YuzuFakeSsidForLdn"}; | ||||||
|  | 
 | ||||||
|  |     bool inited{}; | ||||||
|  |     std::mutex packet_mutex; | ||||||
|  |     std::array<LanStation, StationCountMax> stations; | ||||||
|  |     std::array<NodeLatestUpdate, NodeCountMax> nodeChanges{}; | ||||||
|  |     std::array<u8, NodeCountMax> node_last_states{}; | ||||||
|  |     std::unordered_map<MacAddress, NetworkInfo, MACAddressHash> scan_results{}; | ||||||
|  |     NodeInfo node_info{}; | ||||||
|  |     NetworkInfo network_info{}; | ||||||
|  |     State state{State::None}; | ||||||
|  |     DisconnectReason disconnect_reason{DisconnectReason::None}; | ||||||
|  | 
 | ||||||
|  |     // TODO (flTobi): Should this be an std::set?
 | ||||||
|  |     std::vector<Ipv4Address> connected_clients; | ||||||
|  |     std::optional<Ipv4Address> host_ip = std::nullopt; | ||||||
|  | 
 | ||||||
|  |     LanEventFunc LanEvent; | ||||||
|  | 
 | ||||||
|  |     Network::RoomNetwork& room_network; | ||||||
|  | }; | ||||||
|  | } // namespace Service::LDN
 | ||||||
|  | @ -4,11 +4,13 @@ | ||||||
| #include <memory> | #include <memory> | ||||||
| 
 | 
 | ||||||
| #include "core/core.h" | #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.h" | ||||||
| #include "core/hle/service/ldn/ldn_results.h" | #include "core/hle/service/ldn/ldn_results.h" | ||||||
| #include "core/hle/service/ldn/ldn_types.h" | #include "core/hle/service/ldn/ldn_types.h" | ||||||
| #include "core/internal_network/network.h" | #include "core/internal_network/network.h" | ||||||
| #include "core/internal_network/network_interface.h" | #include "core/internal_network/network_interface.h" | ||||||
|  | #include "network/network.h" | ||||||
| 
 | 
 | ||||||
| // This is defined by synchapi.h and conflicts with ServiceContext::CreateEvent
 | // This is defined by synchapi.h and conflicts with ServiceContext::CreateEvent
 | ||||||
| #undef CreateEvent | #undef CreateEvent | ||||||
|  | @ -105,13 +107,13 @@ class IUserLocalCommunicationService final | ||||||
| public: | public: | ||||||
|     explicit IUserLocalCommunicationService(Core::System& system_) |     explicit IUserLocalCommunicationService(Core::System& system_) | ||||||
|         : ServiceFramework{system_, "IUserLocalCommunicationService", ServiceThreadType::CreateNew}, |         : ServiceFramework{system_, "IUserLocalCommunicationService", ServiceThreadType::CreateNew}, | ||||||
|           service_context{system, "IUserLocalCommunicationService"}, room_network{ |           service_context{system, "IUserLocalCommunicationService"}, | ||||||
|                                                                          system_.GetRoomNetwork()} { |           room_network{system_.GetRoomNetwork()}, lan_discovery{room_network} { | ||||||
|         // clang-format off
 |         // clang-format off
 | ||||||
|         static const FunctionInfo functions[] = { |         static const FunctionInfo functions[] = { | ||||||
|             {0, &IUserLocalCommunicationService::GetState, "GetState"}, |             {0, &IUserLocalCommunicationService::GetState, "GetState"}, | ||||||
|             {1, &IUserLocalCommunicationService::GetNetworkInfo, "GetNetworkInfo"}, |             {1, &IUserLocalCommunicationService::GetNetworkInfo, "GetNetworkInfo"}, | ||||||
|             {2, nullptr, "GetIpv4Address"}, |             {2, &IUserLocalCommunicationService::GetIpv4Address, "GetIpv4Address"}, | ||||||
|             {3, &IUserLocalCommunicationService::GetDisconnectReason, "GetDisconnectReason"}, |             {3, &IUserLocalCommunicationService::GetDisconnectReason, "GetDisconnectReason"}, | ||||||
|             {4, &IUserLocalCommunicationService::GetSecurityParameter, "GetSecurityParameter"}, |             {4, &IUserLocalCommunicationService::GetSecurityParameter, "GetSecurityParameter"}, | ||||||
|             {5, &IUserLocalCommunicationService::GetNetworkConfig, "GetNetworkConfig"}, |             {5, &IUserLocalCommunicationService::GetNetworkConfig, "GetNetworkConfig"}, | ||||||
|  | @ -119,7 +121,7 @@ public: | ||||||
|             {101, &IUserLocalCommunicationService::GetNetworkInfoLatestUpdate, "GetNetworkInfoLatestUpdate"}, |             {101, &IUserLocalCommunicationService::GetNetworkInfoLatestUpdate, "GetNetworkInfoLatestUpdate"}, | ||||||
|             {102, &IUserLocalCommunicationService::Scan, "Scan"}, |             {102, &IUserLocalCommunicationService::Scan, "Scan"}, | ||||||
|             {103, &IUserLocalCommunicationService::ScanPrivate, "ScanPrivate"}, |             {103, &IUserLocalCommunicationService::ScanPrivate, "ScanPrivate"}, | ||||||
|             {104, nullptr, "SetWirelessControllerRestriction"}, |             {104, &IUserLocalCommunicationService::SetWirelessControllerRestriction, "SetWirelessControllerRestriction"}, | ||||||
|             {200, &IUserLocalCommunicationService::OpenAccessPoint, "OpenAccessPoint"}, |             {200, &IUserLocalCommunicationService::OpenAccessPoint, "OpenAccessPoint"}, | ||||||
|             {201, &IUserLocalCommunicationService::CloseAccessPoint, "CloseAccessPoint"}, |             {201, &IUserLocalCommunicationService::CloseAccessPoint, "CloseAccessPoint"}, | ||||||
|             {202, &IUserLocalCommunicationService::CreateNetwork, "CreateNetwork"}, |             {202, &IUserLocalCommunicationService::CreateNetwork, "CreateNetwork"}, | ||||||
|  | @ -148,16 +150,30 @@ public: | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     ~IUserLocalCommunicationService() { |     ~IUserLocalCommunicationService() { | ||||||
|  |         if (is_initialized) { | ||||||
|  |             if (auto room_member = room_network.GetRoomMember().lock()) { | ||||||
|  |                 room_member->Unbind(ldn_packet_received); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         service_context.CloseEvent(state_change_event); |         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() { |     void OnEventFired() { | ||||||
|         state_change_event->GetWritableEvent().Signal(); |         state_change_event->GetWritableEvent().Signal(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void GetState(Kernel::HLERequestContext& ctx) { |     void GetState(Kernel::HLERequestContext& ctx) { | ||||||
|         State state = State::Error; |         State state = State::Error; | ||||||
|         LOG_WARNING(Service_LDN, "(STUBBED) called, state = {}", state); | 
 | ||||||
|  |         if (is_initialized) { | ||||||
|  |             state = lan_discovery.GetState(); | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         IPC::ResponseBuilder rb{ctx, 3}; |         IPC::ResponseBuilder rb{ctx, 3}; | ||||||
|         rb.Push(ResultSuccess); |         rb.Push(ResultSuccess); | ||||||
|  | @ -175,7 +191,7 @@ public: | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         NetworkInfo network_info{}; |         NetworkInfo network_info{}; | ||||||
|         const auto rc = ResultSuccess; |         const auto rc = lan_discovery.GetNetworkInfo(network_info); | ||||||
|         if (rc.IsError()) { |         if (rc.IsError()) { | ||||||
|             LOG_ERROR(Service_LDN, "NetworkInfo is not valid {}", rc.raw); |             LOG_ERROR(Service_LDN, "NetworkInfo is not valid {}", rc.raw); | ||||||
|             IPC::ResponseBuilder rb{ctx, 2}; |             IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|  | @ -183,28 +199,52 @@ public: | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         LOG_WARNING(Service_LDN, "(STUBBED) called, ssid='{}', nodes={}", |  | ||||||
|                     network_info.common.ssid.GetStringValue(), network_info.ldn.node_count); |  | ||||||
| 
 |  | ||||||
|         ctx.WriteBuffer<NetworkInfo>(network_info); |         ctx.WriteBuffer<NetworkInfo>(network_info); | ||||||
|         IPC::ResponseBuilder rb{ctx, 2}; |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|         rb.Push(rc); |         rb.Push(ResultSuccess); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     void GetIpv4Address(Kernel::HLERequestContext& ctx) { | ||||||
|  |         LOG_CRITICAL(Service_LDN, "called"); | ||||||
|  | 
 | ||||||
|  |         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) { |     void GetDisconnectReason(Kernel::HLERequestContext& ctx) { | ||||||
|         const auto disconnect_reason = DisconnectReason::None; |  | ||||||
| 
 |  | ||||||
|         LOG_WARNING(Service_LDN, "(STUBBED) called, disconnect_reason={}", disconnect_reason); |  | ||||||
| 
 |  | ||||||
|         IPC::ResponseBuilder rb{ctx, 3}; |         IPC::ResponseBuilder rb{ctx, 3}; | ||||||
|         rb.Push(ResultSuccess); |         rb.Push(ResultSuccess); | ||||||
|         rb.PushEnum(disconnect_reason); |         rb.PushEnum(lan_discovery.GetDisconnectReason()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void GetSecurityParameter(Kernel::HLERequestContext& ctx) { |     void GetSecurityParameter(Kernel::HLERequestContext& ctx) { | ||||||
|         SecurityParameter security_parameter{}; |         SecurityParameter security_parameter{}; | ||||||
|         NetworkInfo info{}; |         NetworkInfo info{}; | ||||||
|         const Result rc = ResultSuccess; |         const Result rc = lan_discovery.GetNetworkInfo(info); | ||||||
| 
 | 
 | ||||||
|         if (rc.IsError()) { |         if (rc.IsError()) { | ||||||
|             LOG_ERROR(Service_LDN, "NetworkInfo is not valid {}", rc.raw); |             LOG_ERROR(Service_LDN, "NetworkInfo is not valid {}", rc.raw); | ||||||
|  | @ -217,8 +257,6 @@ public: | ||||||
|         std::memcpy(security_parameter.data.data(), info.ldn.security_parameter.data(), |         std::memcpy(security_parameter.data.data(), info.ldn.security_parameter.data(), | ||||||
|                     sizeof(SecurityParameter::data)); |                     sizeof(SecurityParameter::data)); | ||||||
| 
 | 
 | ||||||
|         LOG_WARNING(Service_LDN, "(STUBBED) called"); |  | ||||||
| 
 |  | ||||||
|         IPC::ResponseBuilder rb{ctx, 10}; |         IPC::ResponseBuilder rb{ctx, 10}; | ||||||
|         rb.Push(rc); |         rb.Push(rc); | ||||||
|         rb.PushRaw<SecurityParameter>(security_parameter); |         rb.PushRaw<SecurityParameter>(security_parameter); | ||||||
|  | @ -227,7 +265,7 @@ public: | ||||||
|     void GetNetworkConfig(Kernel::HLERequestContext& ctx) { |     void GetNetworkConfig(Kernel::HLERequestContext& ctx) { | ||||||
|         NetworkConfig config{}; |         NetworkConfig config{}; | ||||||
|         NetworkInfo info{}; |         NetworkInfo info{}; | ||||||
|         const Result rc = ResultSuccess; |         const Result rc = lan_discovery.GetNetworkInfo(info); | ||||||
| 
 | 
 | ||||||
|         if (rc.IsError()) { |         if (rc.IsError()) { | ||||||
|             LOG_ERROR(Service_LDN, "NetworkConfig is not valid {}", rc.raw); |             LOG_ERROR(Service_LDN, "NetworkConfig is not valid {}", rc.raw); | ||||||
|  | @ -241,12 +279,6 @@ public: | ||||||
|         config.node_count_max = info.ldn.node_count_max; |         config.node_count_max = info.ldn.node_count_max; | ||||||
|         config.local_communication_version = info.ldn.nodes[0].local_communication_version; |         config.local_communication_version = info.ldn.nodes[0].local_communication_version; | ||||||
| 
 | 
 | ||||||
|         LOG_WARNING(Service_LDN, |  | ||||||
|                     "(STUBBED) called, intent_id={}/{}, channel={}, node_count_max={}, " |  | ||||||
|                     "local_communication_version={}", |  | ||||||
|                     config.intent_id.local_communication_id, config.intent_id.scene_id, |  | ||||||
|                     config.channel, config.node_count_max, config.local_communication_version); |  | ||||||
| 
 |  | ||||||
|         IPC::ResponseBuilder rb{ctx, 10}; |         IPC::ResponseBuilder rb{ctx, 10}; | ||||||
|         rb.Push(rc); |         rb.Push(rc); | ||||||
|         rb.PushRaw<NetworkConfig>(config); |         rb.PushRaw<NetworkConfig>(config); | ||||||
|  | @ -265,17 +297,17 @@ public: | ||||||
|         const std::size_t node_buffer_count = ctx.GetWriteBufferSize(1) / sizeof(NodeLatestUpdate); |         const std::size_t node_buffer_count = ctx.GetWriteBufferSize(1) / sizeof(NodeLatestUpdate); | ||||||
| 
 | 
 | ||||||
|         if (node_buffer_count == 0 || network_buffer_size != sizeof(NetworkInfo)) { |         if (node_buffer_count == 0 || network_buffer_size != sizeof(NetworkInfo)) { | ||||||
|             LOG_ERROR(Service_LDN, "Invalid buffer size {}, {}", network_buffer_size, |             LOG_ERROR(Service_LDN, "Invalid buffer, size = {}, count = {}", network_buffer_size, | ||||||
|                       node_buffer_count); |                       node_buffer_count); | ||||||
|             IPC::ResponseBuilder rb{ctx, 2}; |             IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|             rb.Push(ResultBadInput); |             rb.Push(ResultBadInput); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         NetworkInfo info; |         NetworkInfo info{}; | ||||||
|         std::vector<NodeLatestUpdate> latest_update(node_buffer_count); |         std::vector<NodeLatestUpdate> latest_update(node_buffer_count); | ||||||
| 
 | 
 | ||||||
|         const auto rc = ResultSuccess; |         const auto rc = lan_discovery.GetNetworkInfo(info, latest_update, latest_update.size()); | ||||||
|         if (rc.IsError()) { |         if (rc.IsError()) { | ||||||
|             LOG_ERROR(Service_LDN, "NetworkInfo is not valid {}", rc.raw); |             LOG_ERROR(Service_LDN, "NetworkInfo is not valid {}", rc.raw); | ||||||
|             IPC::ResponseBuilder rb{ctx, 2}; |             IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|  | @ -283,9 +315,6 @@ public: | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         LOG_WARNING(Service_LDN, "(STUBBED) called, ssid='{}', nodes={}", |  | ||||||
|                     info.common.ssid.GetStringValue(), info.ldn.node_count); |  | ||||||
| 
 |  | ||||||
|         ctx.WriteBuffer(info, 0); |         ctx.WriteBuffer(info, 0); | ||||||
|         ctx.WriteBuffer(latest_update, 1); |         ctx.WriteBuffer(latest_update, 1); | ||||||
| 
 | 
 | ||||||
|  | @ -317,92 +346,78 @@ public: | ||||||
| 
 | 
 | ||||||
|         u16 count = 0; |         u16 count = 0; | ||||||
|         std::vector<NetworkInfo> network_infos(network_info_size); |         std::vector<NetworkInfo> network_infos(network_info_size); | ||||||
|  |         Result rc = lan_discovery.Scan(network_infos, count, scan_filter); | ||||||
| 
 | 
 | ||||||
|         LOG_WARNING(Service_LDN, |         LOG_INFO(Service_LDN, | ||||||
|                     "(STUBBED) called, channel={}, filter_scan_flag={}, filter_network_type={}", |                  "called, channel={}, filter_scan_flag={}, filter_network_type={}, is_private={}", | ||||||
|                     channel, scan_filter.flag, scan_filter.network_type); |                  channel, scan_filter.flag, scan_filter.network_type, is_private); | ||||||
| 
 | 
 | ||||||
|         ctx.WriteBuffer(network_infos); |         ctx.WriteBuffer(network_infos); | ||||||
| 
 | 
 | ||||||
|         IPC::ResponseBuilder rb{ctx, 3}; |         IPC::ResponseBuilder rb{ctx, 3}; | ||||||
|         rb.Push(ResultSuccess); |         rb.Push(rc); | ||||||
|         rb.Push<u32>(count); |         rb.Push<u32>(count); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void OpenAccessPoint(Kernel::HLERequestContext& ctx) { |     void SetWirelessControllerRestriction(Kernel::HLERequestContext& ctx) { | ||||||
|         LOG_WARNING(Service_LDN, "(STUBBED) called"); |         LOG_WARNING(Service_LDN, "(STUBBED) called"); | ||||||
| 
 | 
 | ||||||
|         IPC::ResponseBuilder rb{ctx, 2}; |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|         rb.Push(ResultSuccess); |         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) { |     void CloseAccessPoint(Kernel::HLERequestContext& ctx) { | ||||||
|         LOG_WARNING(Service_LDN, "(STUBBED) called"); |         LOG_INFO(Service_LDN, "called"); | ||||||
| 
 | 
 | ||||||
|         IPC::ResponseBuilder rb{ctx, 2}; |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|         rb.Push(ResultSuccess); |         rb.Push(lan_discovery.CloseAccessPoint()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void CreateNetwork(Kernel::HLERequestContext& ctx) { |     void CreateNetwork(Kernel::HLERequestContext& ctx) { | ||||||
|         IPC::RequestParser rp{ctx}; |         LOG_INFO(Service_LDN, "called"); | ||||||
|         struct Parameters { |  | ||||||
|             SecurityConfig security_config; |  | ||||||
|             UserConfig user_config; |  | ||||||
|             INSERT_PADDING_WORDS_NOINIT(1); |  | ||||||
|             NetworkConfig network_config; |  | ||||||
|         }; |  | ||||||
|         static_assert(sizeof(Parameters) == 0x98, "Parameters has incorrect size."); |  | ||||||
| 
 | 
 | ||||||
|         const auto parameters{rp.PopRaw<Parameters>()}; |         CreateNetworkImpl(ctx); | ||||||
| 
 |  | ||||||
|         LOG_WARNING(Service_LDN, |  | ||||||
|                     "(STUBBED) called, passphrase_size={}, security_mode={}, " |  | ||||||
|                     "local_communication_version={}", |  | ||||||
|                     parameters.security_config.passphrase_size, |  | ||||||
|                     parameters.security_config.security_mode, |  | ||||||
|                     parameters.network_config.local_communication_version); |  | ||||||
| 
 |  | ||||||
|         IPC::ResponseBuilder rb{ctx, 2}; |  | ||||||
|         rb.Push(ResultSuccess); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void CreateNetworkPrivate(Kernel::HLERequestContext& 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}; |         IPC::RequestParser rp{ctx}; | ||||||
|         struct Parameters { |  | ||||||
|             SecurityConfig security_config; |  | ||||||
|             SecurityParameter security_parameter; |  | ||||||
|             UserConfig user_config; |  | ||||||
|             NetworkConfig network_config; |  | ||||||
|         }; |  | ||||||
|         static_assert(sizeof(Parameters) == 0xB8, "Parameters has incorrect size."); |  | ||||||
| 
 | 
 | ||||||
|         const auto parameters{rp.PopRaw<Parameters>()}; |         const auto security_config{rp.PopRaw<SecurityConfig>()}; | ||||||
| 
 |         [[maybe_unused]] const auto security_parameter{is_private ? rp.PopRaw<SecurityParameter>() | ||||||
|         LOG_WARNING(Service_LDN, |                                                                   : SecurityParameter{}}; | ||||||
|                     "(STUBBED) called, passphrase_size={}, security_mode={}, " |         const auto user_config{rp.PopRaw<UserConfig>()}; | ||||||
|                     "local_communication_version={}", |         rp.Pop<u32>(); // Padding
 | ||||||
|                     parameters.security_config.passphrase_size, |         const auto network_Config{rp.PopRaw<NetworkConfig>()}; | ||||||
|                     parameters.security_config.security_mode, |  | ||||||
|                     parameters.network_config.local_communication_version); |  | ||||||
| 
 | 
 | ||||||
|         IPC::ResponseBuilder rb{ctx, 2}; |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|         rb.Push(ResultSuccess); |         rb.Push(lan_discovery.CreateNetwork(security_config, user_config, network_Config)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void DestroyNetwork(Kernel::HLERequestContext& ctx) { |     void DestroyNetwork(Kernel::HLERequestContext& ctx) { | ||||||
|         LOG_WARNING(Service_LDN, "(STUBBED) called"); |         LOG_INFO(Service_LDN, "called"); | ||||||
| 
 | 
 | ||||||
|         IPC::ResponseBuilder rb{ctx, 2}; |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|         rb.Push(ResultSuccess); |         rb.Push(lan_discovery.DestroyNetwork()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void SetAdvertiseData(Kernel::HLERequestContext& ctx) { |     void SetAdvertiseData(Kernel::HLERequestContext& ctx) { | ||||||
|         std::vector<u8> read_buffer = ctx.ReadBuffer(); |         std::vector<u8> read_buffer = ctx.ReadBuffer(); | ||||||
| 
 | 
 | ||||||
|         LOG_WARNING(Service_LDN, "(STUBBED) called, size {}", read_buffer.size()); |  | ||||||
| 
 |  | ||||||
|         IPC::ResponseBuilder rb{ctx, 2}; |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|         rb.Push(ResultSuccess); |         rb.Push(lan_discovery.SetAdvertiseData(read_buffer)); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void SetStationAcceptPolicy(Kernel::HLERequestContext& ctx) { |     void SetStationAcceptPolicy(Kernel::HLERequestContext& ctx) { | ||||||
|  | @ -420,17 +435,17 @@ public: | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void OpenStation(Kernel::HLERequestContext& ctx) { |     void OpenStation(Kernel::HLERequestContext& ctx) { | ||||||
|         LOG_WARNING(Service_LDN, "(STUBBED) called"); |         LOG_INFO(Service_LDN, "called"); | ||||||
| 
 | 
 | ||||||
|         IPC::ResponseBuilder rb{ctx, 2}; |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|         rb.Push(ResultSuccess); |         rb.Push(lan_discovery.OpenStation()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void CloseStation(Kernel::HLERequestContext& ctx) { |     void CloseStation(Kernel::HLERequestContext& ctx) { | ||||||
|         LOG_WARNING(Service_LDN, "(STUBBED) called"); |         LOG_INFO(Service_LDN, "called"); | ||||||
| 
 | 
 | ||||||
|         IPC::ResponseBuilder rb{ctx, 2}; |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|         rb.Push(ResultSuccess); |         rb.Push(lan_discovery.CloseStation()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void Connect(Kernel::HLERequestContext& ctx) { |     void Connect(Kernel::HLERequestContext& ctx) { | ||||||
|  | @ -445,16 +460,13 @@ public: | ||||||
| 
 | 
 | ||||||
|         const auto parameters{rp.PopRaw<Parameters>()}; |         const auto parameters{rp.PopRaw<Parameters>()}; | ||||||
| 
 | 
 | ||||||
|         LOG_WARNING(Service_LDN, |         LOG_INFO(Service_LDN, | ||||||
|                     "(STUBBED) called, passphrase_size={}, security_mode={}, " |                  "called, passphrase_size={}, security_mode={}, " | ||||||
|                  "local_communication_version={}", |                  "local_communication_version={}", | ||||||
|                  parameters.security_config.passphrase_size, |                  parameters.security_config.passphrase_size, | ||||||
|                     parameters.security_config.security_mode, |                  parameters.security_config.security_mode, parameters.local_communication_version); | ||||||
|                     parameters.local_communication_version); |  | ||||||
| 
 | 
 | ||||||
|         const std::vector<u8> read_buffer = ctx.ReadBuffer(); |         const std::vector<u8> read_buffer = ctx.ReadBuffer(); | ||||||
|         NetworkInfo network_info{}; |  | ||||||
| 
 |  | ||||||
|         if (read_buffer.size() != sizeof(NetworkInfo)) { |         if (read_buffer.size() != sizeof(NetworkInfo)) { | ||||||
|             LOG_ERROR(Frontend, "NetworkInfo doesn't match read_buffer size!"); |             LOG_ERROR(Frontend, "NetworkInfo doesn't match read_buffer size!"); | ||||||
|             IPC::ResponseBuilder rb{ctx, 2}; |             IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|  | @ -462,40 +474,47 @@ public: | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         NetworkInfo network_info{}; | ||||||
|         std::memcpy(&network_info, read_buffer.data(), read_buffer.size()); |         std::memcpy(&network_info, read_buffer.data(), read_buffer.size()); | ||||||
| 
 | 
 | ||||||
|         IPC::ResponseBuilder rb{ctx, 2}; |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|         rb.Push(ResultSuccess); |         rb.Push(lan_discovery.Connect(network_info, parameters.user_config, | ||||||
|  |                                       static_cast<u16>(parameters.local_communication_version))); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void Disconnect(Kernel::HLERequestContext& ctx) { |     void Disconnect(Kernel::HLERequestContext& ctx) { | ||||||
|         LOG_WARNING(Service_LDN, "(STUBBED) called"); |         LOG_INFO(Service_LDN, "called"); | ||||||
| 
 | 
 | ||||||
|         IPC::ResponseBuilder rb{ctx, 2}; |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|         rb.Push(ResultSuccess); |         rb.Push(lan_discovery.Disconnect()); | ||||||
|     } |     } | ||||||
|     void Initialize(Kernel::HLERequestContext& ctx) { |  | ||||||
|         LOG_WARNING(Service_LDN, "(STUBBED) called"); |  | ||||||
| 
 | 
 | ||||||
|  |     void Initialize(Kernel::HLERequestContext& ctx) { | ||||||
|         const auto rc = InitializeImpl(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}; |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|         rb.Push(rc); |         rb.Push(rc); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void Finalize(Kernel::HLERequestContext& ctx) { |     void Finalize(Kernel::HLERequestContext& ctx) { | ||||||
|         LOG_WARNING(Service_LDN, "(STUBBED) called"); |         if (auto room_member = room_network.GetRoomMember().lock()) { | ||||||
|  |             room_member->Unbind(ldn_packet_received); | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         is_initialized = false; |         is_initialized = false; | ||||||
| 
 | 
 | ||||||
|         IPC::ResponseBuilder rb{ctx, 2}; |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|         rb.Push(ResultSuccess); |         rb.Push(lan_discovery.Finalize()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void Initialize2(Kernel::HLERequestContext& ctx) { |     void Initialize2(Kernel::HLERequestContext& ctx) { | ||||||
|         LOG_WARNING(Service_LDN, "(STUBBED) called"); |  | ||||||
| 
 |  | ||||||
|         const auto rc = InitializeImpl(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}; |         IPC::ResponseBuilder rb{ctx, 2}; | ||||||
|         rb.Push(rc); |         rb.Push(rc); | ||||||
|  | @ -508,14 +527,26 @@ public: | ||||||
|             return ResultAirplaneModeEnabled; |             return ResultAirplaneModeEnabled; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         is_initialized = true; |         if (auto room_member = room_network.GetRoomMember().lock()) { | ||||||
|         // TODO (flTobi): Change this to ResultSuccess when LDN is fully implemented
 |             ldn_packet_received = room_member->BindOnLdnPacketReceived( | ||||||
|  |                 [this](const Network::LDNPacket& packet) { OnLDNPacketReceived(packet); }); | ||||||
|  |         } else { | ||||||
|  |             LOG_ERROR(Service_LDN, "Couldn't bind callback!"); | ||||||
|             return ResultAirplaneModeEnabled; |             return ResultAirplaneModeEnabled; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         lan_discovery.Initialize([&]() { OnEventFired(); }); | ||||||
|  |         is_initialized = true; | ||||||
|  |         return ResultSuccess; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     KernelHelpers::ServiceContext service_context; |     KernelHelpers::ServiceContext service_context; | ||||||
|     Kernel::KEvent* state_change_event; |     Kernel::KEvent* state_change_event; | ||||||
|     Network::RoomNetwork& room_network; |     Network::RoomNetwork& room_network; | ||||||
|  |     LANDiscovery lan_discovery; | ||||||
|  | 
 | ||||||
|  |     // Callback identifier for the OnLDNPacketReceived event.
 | ||||||
|  |     Network::RoomMember::CallbackHandle<Network::LDNPacket> ldn_packet_received; | ||||||
| 
 | 
 | ||||||
|     bool is_initialized{}; |     bool is_initialized{}; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -31,6 +31,14 @@ enum class NodeStateChange : u8 { | ||||||
|     DisconnectAndConnect, |     DisconnectAndConnect, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | inline NodeStateChange operator|(NodeStateChange a, NodeStateChange b) { | ||||||
|  |     return static_cast<NodeStateChange>(static_cast<u8>(a) | static_cast<u8>(b)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | inline NodeStateChange operator|=(NodeStateChange& a, NodeStateChange b) { | ||||||
|  |     return a = a | b; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| enum class ScanFilterFlag : u32 { | enum class ScanFilterFlag : u32 { | ||||||
|     None = 0, |     None = 0, | ||||||
|     LocalCommunicationId = 1 << 0, |     LocalCommunicationId = 1 << 0, | ||||||
|  | @ -100,13 +108,13 @@ enum class AcceptPolicy : u8 { | ||||||
| 
 | 
 | ||||||
| enum class WifiChannel : s16 { | enum class WifiChannel : s16 { | ||||||
|     Default = 0, |     Default = 0, | ||||||
|     wifi24_1 = 1, |     Wifi24_1 = 1, | ||||||
|     wifi24_6 = 6, |     Wifi24_6 = 6, | ||||||
|     wifi24_11 = 11, |     Wifi24_11 = 11, | ||||||
|     wifi50_36 = 36, |     Wifi50_36 = 36, | ||||||
|     wifi50_40 = 40, |     Wifi50_40 = 40, | ||||||
|     wifi50_44 = 44, |     Wifi50_44 = 44, | ||||||
|     wifi50_48 = 48, |     Wifi50_48 = 48, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| enum class LinkLevel : s8 { | enum class LinkLevel : s8 { | ||||||
|  | @ -116,6 +124,11 @@ enum class LinkLevel : s8 { | ||||||
|     Excellent, |     Excellent, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | enum class NodeStatus : u8 { | ||||||
|  |     Disconnected, | ||||||
|  |     Connected, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| struct NodeLatestUpdate { | struct NodeLatestUpdate { | ||||||
|     NodeStateChange state_change; |     NodeStateChange state_change; | ||||||
|     INSERT_PADDING_BYTES(0x7); // Unknown
 |     INSERT_PADDING_BYTES(0x7); // Unknown
 | ||||||
|  | @ -159,19 +172,14 @@ struct Ssid { | ||||||
|     std::string GetStringValue() const { |     std::string GetStringValue() const { | ||||||
|         return std::string(raw.data()); |         return std::string(raw.data()); | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     bool operator==(const Ssid& b) const { | ||||||
|  |         return (length == b.length) && (std::memcmp(raw.data(), b.raw.data(), length) == 0); | ||||||
|  |     } | ||||||
| }; | }; | ||||||
| static_assert(sizeof(Ssid) == 0x22, "Ssid is an invalid size"); | static_assert(sizeof(Ssid) == 0x22, "Ssid is an invalid size"); | ||||||
| 
 | 
 | ||||||
| struct Ipv4Address { | using Ipv4Address = std::array<u8, 4>; | ||||||
|     union { |  | ||||||
|         u32 raw{}; |  | ||||||
|         std::array<u8, 4> bytes; |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     std::string GetStringValue() const { |  | ||||||
|         return fmt::format("{}.{}.{}.{}", bytes[3], bytes[2], bytes[1], bytes[0]); |  | ||||||
|     } |  | ||||||
| }; |  | ||||||
| static_assert(sizeof(Ipv4Address) == 0x4, "Ipv4Address is an invalid size"); | static_assert(sizeof(Ipv4Address) == 0x4, "Ipv4Address is an invalid size"); | ||||||
| 
 | 
 | ||||||
| struct MacAddress { | struct MacAddress { | ||||||
|  | @ -181,6 +189,14 @@ struct MacAddress { | ||||||
| }; | }; | ||||||
| static_assert(sizeof(MacAddress) == 0x6, "MacAddress is an invalid size"); | static_assert(sizeof(MacAddress) == 0x6, "MacAddress is an invalid size"); | ||||||
| 
 | 
 | ||||||
|  | struct MACAddressHash { | ||||||
|  |     size_t operator()(const MacAddress& address) const { | ||||||
|  |         u64 value{}; | ||||||
|  |         std::memcpy(&value, address.raw.data(), sizeof(address.raw)); | ||||||
|  |         return value; | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  | 
 | ||||||
| struct ScanFilter { | struct ScanFilter { | ||||||
|     NetworkId network_id; |     NetworkId network_id; | ||||||
|     NetworkType network_type; |     NetworkType network_type; | ||||||
|  |  | ||||||
|  | @ -6,6 +6,7 @@ | ||||||
| 
 | 
 | ||||||
| #include "common/assert.h" | #include "common/assert.h" | ||||||
| #include "common/logging/log.h" | #include "common/logging/log.h" | ||||||
|  | #include "common/zstd_compression.h" | ||||||
| #include "core/internal_network/network.h" | #include "core/internal_network/network.h" | ||||||
| #include "core/internal_network/network_interface.h" | #include "core/internal_network/network_interface.h" | ||||||
| #include "core/internal_network/socket_proxy.h" | #include "core/internal_network/socket_proxy.h" | ||||||
|  | @ -32,8 +33,11 @@ void ProxySocket::HandleProxyPacket(const ProxyPacket& packet) { | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     auto decompressed = packet; | ||||||
|  |     decompressed.data = Common::Compression::DecompressDataZSTD(packet.data); | ||||||
|  | 
 | ||||||
|     std::lock_guard guard(packets_mutex); |     std::lock_guard guard(packets_mutex); | ||||||
|     received_packets.push(packet); |     received_packets.push(decompressed); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| template <typename T> | template <typename T> | ||||||
|  | @ -185,6 +189,8 @@ std::pair<s32, Errno> ProxySocket::Send(const std::vector<u8>& message, int flag | ||||||
| void ProxySocket::SendPacket(ProxyPacket& packet) { | void ProxySocket::SendPacket(ProxyPacket& packet) { | ||||||
|     if (auto room_member = room_network.GetRoomMember().lock()) { |     if (auto room_member = room_network.GetRoomMember().lock()) { | ||||||
|         if (room_member->IsConnected()) { |         if (room_member->IsConnected()) { | ||||||
|  |             packet.data = Common::Compression::CompressDataZSTDDefault(packet.data.data(), | ||||||
|  |                                                                        packet.data.size()); | ||||||
|             room_member->SendProxyPacket(packet); |             room_member->SendProxyPacket(packet); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -76,7 +76,8 @@ static constexpr char BanListMagic[] = "YuzuRoom-BanList-1"; | ||||||
| static constexpr char token_delimiter{':'}; | static constexpr char token_delimiter{':'}; | ||||||
| 
 | 
 | ||||||
| static void PadToken(std::string& token) { | static void PadToken(std::string& token) { | ||||||
|     while (token.size() % 4 != 0) { |     const auto remainder = token.size() % 3; | ||||||
|  |     for (size_t i = 0; i < (3 - remainder); i++) { | ||||||
|         token.push_back('='); |         token.push_back('='); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -211,6 +211,12 @@ public: | ||||||
|      */ |      */ | ||||||
|     void HandleProxyPacket(const ENetEvent* event); |     void HandleProxyPacket(const ENetEvent* event); | ||||||
| 
 | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Broadcasts this packet to all members except the sender. | ||||||
|  |      * @param event The ENet event containing the data | ||||||
|  |      */ | ||||||
|  |     void HandleLdnPacket(const ENetEvent* event); | ||||||
|  | 
 | ||||||
|     /**
 |     /**
 | ||||||
|      * Extracts a chat entry from a received ENet packet and adds it to the chat queue. |      * Extracts a chat entry from a received ENet packet and adds it to the chat queue. | ||||||
|      * @param event The ENet event that was received. |      * @param event The ENet event that was received. | ||||||
|  | @ -247,6 +253,9 @@ void Room::RoomImpl::ServerLoop() { | ||||||
|                 case IdProxyPacket: |                 case IdProxyPacket: | ||||||
|                     HandleProxyPacket(&event); |                     HandleProxyPacket(&event); | ||||||
|                     break; |                     break; | ||||||
|  |                 case IdLdnPacket: | ||||||
|  |                     HandleLdnPacket(&event); | ||||||
|  |                     break; | ||||||
|                 case IdChatMessage: |                 case IdChatMessage: | ||||||
|                     HandleChatPacket(&event); |                     HandleChatPacket(&event); | ||||||
|                     break; |                     break; | ||||||
|  | @ -861,6 +870,60 @@ void Room::RoomImpl::HandleProxyPacket(const ENetEvent* event) { | ||||||
|     enet_host_flush(server); |     enet_host_flush(server); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void Room::RoomImpl::HandleLdnPacket(const ENetEvent* event) { | ||||||
|  |     Packet in_packet; | ||||||
|  |     in_packet.Append(event->packet->data, event->packet->dataLength); | ||||||
|  | 
 | ||||||
|  |     in_packet.IgnoreBytes(sizeof(u8)); // Message type
 | ||||||
|  | 
 | ||||||
|  |     in_packet.IgnoreBytes(sizeof(u8));          // LAN packet type
 | ||||||
|  |     in_packet.IgnoreBytes(sizeof(IPv4Address)); // Local IP
 | ||||||
|  | 
 | ||||||
|  |     IPv4Address remote_ip; | ||||||
|  |     in_packet.Read(remote_ip); // Remote IP
 | ||||||
|  | 
 | ||||||
|  |     bool broadcast; | ||||||
|  |     in_packet.Read(broadcast); // Broadcast
 | ||||||
|  | 
 | ||||||
|  |     Packet out_packet; | ||||||
|  |     out_packet.Append(event->packet->data, event->packet->dataLength); | ||||||
|  |     ENetPacket* enet_packet = enet_packet_create(out_packet.GetData(), out_packet.GetDataSize(), | ||||||
|  |                                                  ENET_PACKET_FLAG_RELIABLE); | ||||||
|  | 
 | ||||||
|  |     const auto& destination_address = remote_ip; | ||||||
|  |     if (broadcast) { // Send the data to everyone except the sender
 | ||||||
|  |         std::lock_guard lock(member_mutex); | ||||||
|  |         bool sent_packet = false; | ||||||
|  |         for (const auto& member : members) { | ||||||
|  |             if (member.peer != event->peer) { | ||||||
|  |                 sent_packet = true; | ||||||
|  |                 enet_peer_send(member.peer, 0, enet_packet); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (!sent_packet) { | ||||||
|  |             enet_packet_destroy(enet_packet); | ||||||
|  |         } | ||||||
|  |     } else { | ||||||
|  |         std::lock_guard lock(member_mutex); | ||||||
|  |         auto member = std::find_if(members.begin(), members.end(), | ||||||
|  |                                    [destination_address](const Member& member_entry) -> bool { | ||||||
|  |                                        return member_entry.fake_ip == destination_address; | ||||||
|  |                                    }); | ||||||
|  |         if (member != members.end()) { | ||||||
|  |             enet_peer_send(member->peer, 0, enet_packet); | ||||||
|  |         } else { | ||||||
|  |             LOG_ERROR(Network, | ||||||
|  |                       "Attempting to send to unknown IP address: " | ||||||
|  |                       "{}.{}.{}.{}", | ||||||
|  |                       destination_address[0], destination_address[1], destination_address[2], | ||||||
|  |                       destination_address[3]); | ||||||
|  |             enet_packet_destroy(enet_packet); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     enet_host_flush(server); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void Room::RoomImpl::HandleChatPacket(const ENetEvent* event) { | void Room::RoomImpl::HandleChatPacket(const ENetEvent* event) { | ||||||
|     Packet in_packet; |     Packet in_packet; | ||||||
|     in_packet.Append(event->packet->data, event->packet->dataLength); |     in_packet.Append(event->packet->data, event->packet->dataLength); | ||||||
|  |  | ||||||
|  | @ -40,6 +40,7 @@ enum RoomMessageTypes : u8 { | ||||||
|     IdRoomInformation, |     IdRoomInformation, | ||||||
|     IdSetGameInfo, |     IdSetGameInfo, | ||||||
|     IdProxyPacket, |     IdProxyPacket, | ||||||
|  |     IdLdnPacket, | ||||||
|     IdChatMessage, |     IdChatMessage, | ||||||
|     IdNameCollision, |     IdNameCollision, | ||||||
|     IdIpCollision, |     IdIpCollision, | ||||||
|  |  | ||||||
|  | @ -58,6 +58,7 @@ public: | ||||||
| 
 | 
 | ||||||
|     private: |     private: | ||||||
|         CallbackSet<ProxyPacket> callback_set_proxy_packet; |         CallbackSet<ProxyPacket> callback_set_proxy_packet; | ||||||
|  |         CallbackSet<LDNPacket> callback_set_ldn_packet; | ||||||
|         CallbackSet<ChatEntry> callback_set_chat_messages; |         CallbackSet<ChatEntry> callback_set_chat_messages; | ||||||
|         CallbackSet<StatusMessageEntry> callback_set_status_messages; |         CallbackSet<StatusMessageEntry> callback_set_status_messages; | ||||||
|         CallbackSet<RoomInformation> callback_set_room_information; |         CallbackSet<RoomInformation> callback_set_room_information; | ||||||
|  | @ -107,6 +108,12 @@ public: | ||||||
|      */ |      */ | ||||||
|     void HandleProxyPackets(const ENetEvent* event); |     void HandleProxyPackets(const ENetEvent* event); | ||||||
| 
 | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Extracts an LdnPacket from a received ENet packet. | ||||||
|  |      * @param event The ENet event that was received. | ||||||
|  |      */ | ||||||
|  |     void HandleLdnPackets(const ENetEvent* event); | ||||||
|  | 
 | ||||||
|     /**
 |     /**
 | ||||||
|      * Extracts a chat entry from a received ENet packet and adds it to the chat queue. |      * Extracts a chat entry from a received ENet packet and adds it to the chat queue. | ||||||
|      * @param event The ENet event that was received. |      * @param event The ENet event that was received. | ||||||
|  | @ -166,6 +173,9 @@ void RoomMember::RoomMemberImpl::MemberLoop() { | ||||||
|                 case IdProxyPacket: |                 case IdProxyPacket: | ||||||
|                     HandleProxyPackets(&event); |                     HandleProxyPackets(&event); | ||||||
|                     break; |                     break; | ||||||
|  |                 case IdLdnPacket: | ||||||
|  |                     HandleLdnPackets(&event); | ||||||
|  |                     break; | ||||||
|                 case IdChatMessage: |                 case IdChatMessage: | ||||||
|                     HandleChatPacket(&event); |                     HandleChatPacket(&event); | ||||||
|                     break; |                     break; | ||||||
|  | @ -372,6 +382,27 @@ void RoomMember::RoomMemberImpl::HandleProxyPackets(const ENetEvent* event) { | ||||||
|     Invoke<ProxyPacket>(proxy_packet); |     Invoke<ProxyPacket>(proxy_packet); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void RoomMember::RoomMemberImpl::HandleLdnPackets(const ENetEvent* event) { | ||||||
|  |     LDNPacket ldn_packet{}; | ||||||
|  |     Packet packet; | ||||||
|  |     packet.Append(event->packet->data, event->packet->dataLength); | ||||||
|  | 
 | ||||||
|  |     // Ignore the first byte, which is the message id.
 | ||||||
|  |     packet.IgnoreBytes(sizeof(u8)); // Ignore the message type
 | ||||||
|  | 
 | ||||||
|  |     u8 packet_type; | ||||||
|  |     packet.Read(packet_type); | ||||||
|  |     ldn_packet.type = static_cast<LDNPacketType>(packet_type); | ||||||
|  | 
 | ||||||
|  |     packet.Read(ldn_packet.local_ip); | ||||||
|  |     packet.Read(ldn_packet.remote_ip); | ||||||
|  |     packet.Read(ldn_packet.broadcast); | ||||||
|  | 
 | ||||||
|  |     packet.Read(ldn_packet.data); | ||||||
|  | 
 | ||||||
|  |     Invoke<LDNPacket>(ldn_packet); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void RoomMember::RoomMemberImpl::HandleChatPacket(const ENetEvent* event) { | void RoomMember::RoomMemberImpl::HandleChatPacket(const ENetEvent* event) { | ||||||
|     Packet packet; |     Packet packet; | ||||||
|     packet.Append(event->packet->data, event->packet->dataLength); |     packet.Append(event->packet->data, event->packet->dataLength); | ||||||
|  | @ -449,6 +480,11 @@ RoomMember::RoomMemberImpl::CallbackSet<ProxyPacket>& RoomMember::RoomMemberImpl | ||||||
|     return callback_set_proxy_packet; |     return callback_set_proxy_packet; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | template <> | ||||||
|  | RoomMember::RoomMemberImpl::CallbackSet<LDNPacket>& RoomMember::RoomMemberImpl::Callbacks::Get() { | ||||||
|  |     return callback_set_ldn_packet; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| template <> | template <> | ||||||
| RoomMember::RoomMemberImpl::CallbackSet<RoomMember::State>& | RoomMember::RoomMemberImpl::CallbackSet<RoomMember::State>& | ||||||
| RoomMember::RoomMemberImpl::Callbacks::Get() { | RoomMember::RoomMemberImpl::Callbacks::Get() { | ||||||
|  | @ -607,6 +643,21 @@ void RoomMember::SendProxyPacket(const ProxyPacket& proxy_packet) { | ||||||
|     room_member_impl->Send(std::move(packet)); |     room_member_impl->Send(std::move(packet)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void RoomMember::SendLdnPacket(const LDNPacket& ldn_packet) { | ||||||
|  |     Packet packet; | ||||||
|  |     packet.Write(static_cast<u8>(IdLdnPacket)); | ||||||
|  | 
 | ||||||
|  |     packet.Write(static_cast<u8>(ldn_packet.type)); | ||||||
|  | 
 | ||||||
|  |     packet.Write(ldn_packet.local_ip); | ||||||
|  |     packet.Write(ldn_packet.remote_ip); | ||||||
|  |     packet.Write(ldn_packet.broadcast); | ||||||
|  | 
 | ||||||
|  |     packet.Write(ldn_packet.data); | ||||||
|  | 
 | ||||||
|  |     room_member_impl->Send(std::move(packet)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void RoomMember::SendChatMessage(const std::string& message) { | void RoomMember::SendChatMessage(const std::string& message) { | ||||||
|     Packet packet; |     Packet packet; | ||||||
|     packet.Write(static_cast<u8>(IdChatMessage)); |     packet.Write(static_cast<u8>(IdChatMessage)); | ||||||
|  | @ -663,6 +714,11 @@ RoomMember::CallbackHandle<ProxyPacket> RoomMember::BindOnProxyPacketReceived( | ||||||
|     return room_member_impl->Bind(callback); |     return room_member_impl->Bind(callback); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | RoomMember::CallbackHandle<LDNPacket> RoomMember::BindOnLdnPacketReceived( | ||||||
|  |     std::function<void(const LDNPacket&)> callback) { | ||||||
|  |     return room_member_impl->Bind(callback); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| RoomMember::CallbackHandle<RoomInformation> RoomMember::BindOnRoomInformationChanged( | RoomMember::CallbackHandle<RoomInformation> RoomMember::BindOnRoomInformationChanged( | ||||||
|     std::function<void(const RoomInformation&)> callback) { |     std::function<void(const RoomInformation&)> callback) { | ||||||
|     return room_member_impl->Bind(callback); |     return room_member_impl->Bind(callback); | ||||||
|  | @ -699,6 +755,7 @@ void RoomMember::Leave() { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| template void RoomMember::Unbind(CallbackHandle<ProxyPacket>); | template void RoomMember::Unbind(CallbackHandle<ProxyPacket>); | ||||||
|  | template void RoomMember::Unbind(CallbackHandle<LDNPacket>); | ||||||
| template void RoomMember::Unbind(CallbackHandle<RoomMember::State>); | template void RoomMember::Unbind(CallbackHandle<RoomMember::State>); | ||||||
| template void RoomMember::Unbind(CallbackHandle<RoomMember::Error>); | template void RoomMember::Unbind(CallbackHandle<RoomMember::Error>); | ||||||
| template void RoomMember::Unbind(CallbackHandle<RoomInformation>); | template void RoomMember::Unbind(CallbackHandle<RoomInformation>); | ||||||
|  |  | ||||||
|  | @ -17,7 +17,24 @@ namespace Network { | ||||||
| using AnnounceMultiplayerRoom::GameInfo; | using AnnounceMultiplayerRoom::GameInfo; | ||||||
| using AnnounceMultiplayerRoom::RoomInformation; | using AnnounceMultiplayerRoom::RoomInformation; | ||||||
| 
 | 
 | ||||||
| /// Information about the received WiFi packets.
 | enum class LDNPacketType : u8 { | ||||||
|  |     Scan, | ||||||
|  |     ScanResp, | ||||||
|  |     Connect, | ||||||
|  |     SyncNetwork, | ||||||
|  |     Disconnect, | ||||||
|  |     DestroyNetwork, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct LDNPacket { | ||||||
|  |     LDNPacketType type; | ||||||
|  |     IPv4Address local_ip; | ||||||
|  |     IPv4Address remote_ip; | ||||||
|  |     bool broadcast; | ||||||
|  |     std::vector<u8> data; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /// Information about the received proxy packets.
 | ||||||
| struct ProxyPacket { | struct ProxyPacket { | ||||||
|     SockAddrIn local_endpoint; |     SockAddrIn local_endpoint; | ||||||
|     SockAddrIn remote_endpoint; |     SockAddrIn remote_endpoint; | ||||||
|  | @ -151,6 +168,12 @@ public: | ||||||
|      */ |      */ | ||||||
|     void SendProxyPacket(const ProxyPacket& packet); |     void SendProxyPacket(const ProxyPacket& packet); | ||||||
| 
 | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Sends an LDN packet to the room. | ||||||
|  |      * @param packet The WiFi packet to send. | ||||||
|  |      */ | ||||||
|  |     void SendLdnPacket(const LDNPacket& packet); | ||||||
|  | 
 | ||||||
|     /**
 |     /**
 | ||||||
|      * Sends a chat message to the room. |      * Sends a chat message to the room. | ||||||
|      * @param message The contents of the message. |      * @param message The contents of the message. | ||||||
|  | @ -204,6 +227,16 @@ public: | ||||||
|     CallbackHandle<ProxyPacket> BindOnProxyPacketReceived( |     CallbackHandle<ProxyPacket> BindOnProxyPacketReceived( | ||||||
|         std::function<void(const ProxyPacket&)> callback); |         std::function<void(const ProxyPacket&)> callback); | ||||||
| 
 | 
 | ||||||
|  |     /**
 | ||||||
|  |      * Binds a function to an event that will be triggered every time an LDNPacket is received. | ||||||
|  |      * The function wil be called everytime the event is triggered. | ||||||
|  |      * The callback function must not bind or unbind a function. Doing so will cause a deadlock | ||||||
|  |      * @param callback The function to call | ||||||
|  |      * @return A handle used for removing the function from the registered list | ||||||
|  |      */ | ||||||
|  |     CallbackHandle<LDNPacket> BindOnLdnPacketReceived( | ||||||
|  |         std::function<void(const LDNPacket&)> callback); | ||||||
|  | 
 | ||||||
|     /**
 |     /**
 | ||||||
|      * Binds a function to an event that will be triggered every time the RoomInformation changes. |      * Binds a function to an event that will be triggered every time the RoomInformation changes. | ||||||
|      * The function wil be called every time the event is triggered. |      * The function wil be called every time the event is triggered. | ||||||
|  |  | ||||||
|  | @ -896,8 +896,8 @@ void GMainWindow::InitializeWidgets() { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // TODO (flTobi): Add the widget when multiplayer is fully implemented
 |     // TODO (flTobi): Add the widget when multiplayer is fully implemented
 | ||||||
|     // statusBar()->addPermanentWidget(multiplayer_state->GetStatusText(), 0);
 |     statusBar()->addPermanentWidget(multiplayer_state->GetStatusText(), 0); | ||||||
|     // statusBar()->addPermanentWidget(multiplayer_state->GetStatusIcon(), 0);
 |     statusBar()->addPermanentWidget(multiplayer_state->GetStatusIcon(), 0); | ||||||
| 
 | 
 | ||||||
|     tas_label = new QLabel(); |     tas_label = new QLabel(); | ||||||
|     tas_label->setObjectName(QStringLiteral("TASlabel")); |     tas_label->setObjectName(QStringLiteral("TASlabel")); | ||||||
|  |  | ||||||
|  | @ -120,6 +120,20 @@ | ||||||
|     <addaction name="menu_Reset_Window_Size"/> |     <addaction name="menu_Reset_Window_Size"/> | ||||||
|     <addaction name="menu_View_Debugging"/> |     <addaction name="menu_View_Debugging"/> | ||||||
|    </widget> |    </widget> | ||||||
|  |    <widget class="QMenu" name="menu_Multiplayer"> | ||||||
|  |     <property name="enabled"> | ||||||
|  |      <bool>true</bool> | ||||||
|  |     </property> | ||||||
|  |     <property name="title"> | ||||||
|  |      <string>Multiplayer</string> | ||||||
|  |     </property> | ||||||
|  |     <addaction name="action_View_Lobby"/> | ||||||
|  |     <addaction name="action_Start_Room"/> | ||||||
|  |     <addaction name="action_Connect_To_Room"/> | ||||||
|  |     <addaction name="separator"/> | ||||||
|  |     <addaction name="action_Show_Room"/> | ||||||
|  |     <addaction name="action_Leave_Room"/> | ||||||
|  |    </widget> | ||||||
|    <widget class="QMenu" name="menu_Tools"> |    <widget class="QMenu" name="menu_Tools"> | ||||||
|     <property name="title"> |     <property name="title"> | ||||||
|      <string>&Tools</string> |      <string>&Tools</string> | ||||||
|  |  | ||||||
|  | @ -61,7 +61,10 @@ public: | ||||||
| 
 | 
 | ||||||
|     /// Format the message using the players color
 |     /// Format the message using the players color
 | ||||||
|     QString GetPlayerChatMessage(u16 player) const { |     QString GetPlayerChatMessage(u16 player) const { | ||||||
|         auto color = player_color[player % 16]; |         const bool is_dark_theme = QIcon::themeName().contains(QStringLiteral("dark")) || | ||||||
|  |                                    QIcon::themeName().contains(QStringLiteral("midnight")); | ||||||
|  |         auto color = | ||||||
|  |             is_dark_theme ? player_color_dark[player % 16] : player_color_default[player % 16]; | ||||||
|         QString name; |         QString name; | ||||||
|         if (username.isEmpty() || username == nickname) { |         if (username.isEmpty() || username == nickname) { | ||||||
|             name = nickname; |             name = nickname; | ||||||
|  | @ -84,9 +87,12 @@ public: | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|     static constexpr std::array<const char*, 16> player_color = { |     static constexpr std::array<const char*, 16> player_color_default = { | ||||||
|         {"#0000FF", "#FF0000", "#8A2BE2", "#FF69B4", "#1E90FF", "#008000", "#00FF7F", "#B22222", |         {"#0000FF", "#FF0000", "#8A2BE2", "#FF69B4", "#1E90FF", "#008000", "#00FF7F", "#B22222", | ||||||
|          "#DAA520", "#FF4500", "#2E8B57", "#5F9EA0", "#D2691E", "#9ACD32", "#FF7F50", "FFFF00"}}; |          "#DAA520", "#FF4500", "#2E8B57", "#5F9EA0", "#D2691E", "#9ACD32", "#FF7F50", "#FFFF00"}}; | ||||||
|  |     static constexpr std::array<const char*, 16> player_color_dark = { | ||||||
|  |         {"#559AD1", "#4EC9A8", "#D69D85", "#C6C923", "#B975B5", "#D81F1F", "#7EAE39", "#4F8733", | ||||||
|  |          "#F7CD8A", "#6FCACF", "#CE4897", "#8A2BE2", "#D2691E", "#9ACD32", "#FF7F50", "#152ccd"}}; | ||||||
|     static constexpr char ping_color[] = "#FFFF00"; |     static constexpr char ping_color[] = "#FFFF00"; | ||||||
| 
 | 
 | ||||||
|     QString timestamp; |     QString timestamp; | ||||||
|  |  | ||||||
|  | @ -249,6 +249,7 @@ void MultiplayerState::ShowNotification() { | ||||||
|         return; // Do not show notification if the chat window currently has focus
 |         return; // Do not show notification if the chat window currently has focus
 | ||||||
|     show_notification = true; |     show_notification = true; | ||||||
|     QApplication::alert(nullptr); |     QApplication::alert(nullptr); | ||||||
|  |     QApplication::beep(); | ||||||
|     status_icon->setPixmap(QIcon::fromTheme(QStringLiteral("connected_notification")).pixmap(16)); |     status_icon->setPixmap(QIcon::fromTheme(QStringLiteral("connected_notification")).pixmap(16)); | ||||||
|     status_text->setText(tr("New Messages Received")); |     status_text->setText(tr("New Messages Received")); | ||||||
| } | } | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 FearlessTobi
						FearlessTobi