/* * Copyright (c) 2022, Liav A. * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include namespace Kernel::PCI { NonnullOwnPtr HostBridge::must_create_with_io_access() { PCI::Domain domain { 0, 0, 0xff }; return adopt_own_if_nonnull(new (nothrow) HostBridge(domain)).release_nonnull(); } HostBridge::HostBridge(PCI::Domain const& domain) : HostController(domain) { } static u32 io_address_for_pci_field(BusNumber bus, DeviceNumber device, FunctionNumber function, u8 field) { return 0x80000000u | (bus.value() << 16u) | (device.value() << 11u) | (function.value() << 8u) | (field & 0xfc); } void HostBridge::write8_field(BusNumber bus, DeviceNumber device, FunctionNumber function, u32 field, u8 value) { IO::out32(PCI::address_port, io_address_for_pci_field(bus, device, function, field)); IO::out8(PCI::value_port + (field & 3), value); } void HostBridge::write16_field(BusNumber bus, DeviceNumber device, FunctionNumber function, u32 field, u16 value) { IO::out32(PCI::address_port, io_address_for_pci_field(bus, device, function, field)); IO::out16(PCI::value_port + (field & 2), value); } void HostBridge::write32_field(BusNumber bus, DeviceNumber device, FunctionNumber function, u32 field, u32 value) { IO::out32(PCI::address_port, io_address_for_pci_field(bus, device, function, field)); IO::out32(PCI::value_port, value); } u8 HostBridge::read8_field(BusNumber bus, DeviceNumber device, FunctionNumber function, u32 field) { IO::out32(PCI::address_port, io_address_for_pci_field(bus, device, function, field)); return IO::in8(PCI::value_port + (field & 3)); } u16 HostBridge::read16_field(BusNumber bus, DeviceNumber device, FunctionNumber function, u32 field) { IO::out32(PCI::address_port, io_address_for_pci_field(bus, device, function, field)); return IO::in16(PCI::value_port + (field & 2)); } u32 HostBridge::read32_field(BusNumber bus, DeviceNumber device, FunctionNumber function, u32 field) { IO::out32(PCI::address_port, io_address_for_pci_field(bus, device, function, field)); return IO::in32(PCI::value_port); } }