/* * Copyright (c) 2021, Liav A. * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace Kernel { static Singleton s_the; NetworkingManagement& NetworkingManagement::the() { return *s_the; } bool NetworkingManagement::is_initialized() { return s_the.is_initialized(); } UNMAP_AFTER_INIT NetworkingManagement::NetworkingManagement() { } NonnullRefPtr NetworkingManagement::loopback_adapter() const { return *m_loopback_adapter; } void NetworkingManagement::for_each(Function callback) { MutexLocker locker(m_lock); for (auto& it : m_adapters) callback(it); } RefPtr NetworkingManagement::from_ipv4_address(const IPv4Address& address) const { MutexLocker locker(m_lock); for (auto& adapter : m_adapters) { if (adapter.ipv4_address() == address || adapter.ipv4_broadcast() == address) return adapter; } if (address[0] == 0 && address[1] == 0 && address[2] == 0 && address[3] == 0) return m_loopback_adapter; if (address[0] == 127) return m_loopback_adapter; return {}; } RefPtr NetworkingManagement::lookup_by_name(const StringView& name) const { MutexLocker locker(m_lock); RefPtr found_adapter; for (auto& it : m_adapters) { if (it.name() == name) found_adapter = it; } return found_adapter; } UNMAP_AFTER_INIT RefPtr NetworkingManagement::determine_network_device(PCI::DeviceIdentifier const& device_identifier) const { // Note: This stands for e - "Ethernet", p - "Port" as for PCI bus, "s" for slot as for PCI slot auto name = String::formatted("ep{}s{}", device_identifier.address().bus(), device_identifier.address().device()); VERIFY(!NetworkingManagement::the().lookup_by_name(name)); // TODO: We need some way to to format data into a `KString`. auto interface_name_or_error = KString::try_create(name.view()); if (interface_name_or_error.is_error()) return {}; auto interface_name = interface_name_or_error.release_value(); if (auto candidate = E1000NetworkAdapter::try_to_initialize(device_identifier, move(interface_name)); !candidate.is_null()) return candidate; if (auto candidate = E1000ENetworkAdapter::try_to_initialize(device_identifier, move(interface_name)); !candidate.is_null()) return candidate; if (auto candidate = RTL8139NetworkAdapter::try_to_initialize(device_identifier, move(interface_name)); !candidate.is_null()) return candidate; if (auto candidate = RTL8168NetworkAdapter::try_to_initialize(device_identifier, move(interface_name)); !candidate.is_null()) return candidate; if (auto candidate = NE2000NetworkAdapter::try_to_initialize(device_identifier, move(interface_name)); !candidate.is_null()) return candidate; return {}; } bool NetworkingManagement::initialize() { if (!kernel_command_line().is_physical_networking_disabled()) { PCI::enumerate([&](PCI::DeviceIdentifier const& device_identifier) { // Note: PCI class 2 is the class of Network devices if (device_identifier.class_code().value() != 0x02) return; if (auto adapter = determine_network_device(device_identifier); !adapter.is_null()) m_adapters.append(adapter.release_nonnull()); }); } auto loopback = LoopbackAdapter::try_create(); VERIFY(loopback); m_adapters.append(*loopback); m_loopback_adapter = loopback; return true; } }