1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-07-25 19:57:35 +00:00

Kernel/USB: Add a rudimentary interrogation only EHCI driver

This adds a simple EHCI driver that currently only interrogates the
device and checks if all ports are addressable via associated legacy
controllers (companion controllers), and warns if this is not the case.

This also adds a lot of the other data structures needed for actually
driving the controller, but these are currently not hooked up to
anything.

To test this run with `SERENITY_EXTRA_QEMU_ARGS="--device usb-ehci"`
or the q35 machine type
This commit is contained in:
Hendiadyoin1 2023-12-13 21:59:03 +01:00 committed by Andrew Kaster
parent a768685d16
commit f4bfd0468b
6 changed files with 742 additions and 1 deletions

View file

@ -0,0 +1,83 @@
/*
* Copyright (c) 2023, Leon Albrecht <leon.a@serenityos.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <Kernel/Bus/PCI/API.h>
#include <Kernel/Bus/USB/EHCI/EHCIController.h>
namespace Kernel::USB::EHCI {
ErrorOr<NonnullLockRefPtr<EHCIController>> EHCIController::try_to_initialize(const PCI::DeviceIdentifier& pci_device_identifier)
{
// FIXME: This assumes the BIOS left us a physical region for the controller
u64 pci_bar_value = PCI::get_BAR(pci_device_identifier, SpaceBaseAddressRegister);
auto pci_bar_space_type = PCI::get_BAR_space_type(pci_bar_value);
if (pci_bar_space_type == PCI::BARSpaceType::Memory64BitSpace) {
u64 next_pci_bar_value = PCI::get_BAR(pci_device_identifier, static_cast<PCI::HeaderType0BaseRegister>(to_underlying(SpaceBaseAddressRegister) + 1));
pci_bar_value |= next_pci_bar_value << 32;
}
auto pci_bar_space_size = PCI::get_BAR_space_size(pci_device_identifier, SpaceBaseAddressRegister);
auto register_region = TRY(MM.allocate_kernel_region(PhysicalAddress { pci_bar_value }, pci_bar_space_size, {}, Memory::Region::Access::ReadWrite));
PCI::enable_bus_mastering(pci_device_identifier);
PCI::enable_memory_space(pci_device_identifier);
auto controller = TRY(adopt_nonnull_lock_ref_or_enomem(new (nothrow) EHCIController(pci_device_identifier, move(register_region))));
TRY(controller->initialize());
return controller;
}
EHCIController::EHCIController(PCI::DeviceIdentifier const& pci_device_identifier, NonnullOwnPtr<Memory::Region> register_region)
: PCI::Device(pci_device_identifier)
, m_register_region(move(register_region))
{
m_cap_regs = bit_cast<CapabilityRegisters const*>(m_register_region->vaddr().get());
m_op_regs = bit_cast<OperationalRegisters volatile*>(m_register_region->vaddr().get() + m_cap_regs->capability_length);
}
ErrorOr<void> EHCIController::initialize()
{
dmesgln_pci(*this, "Controller found {} @ {}", PCI::get_hardware_id(device_identifier()), device_identifier().address());
dmesgln_pci(*this, "Version {}.{}", m_cap_regs->interface_version.major, m_cap_regs->interface_version.minor);
u8 n_ports = m_cap_regs->structural_parameters.n_ports;
dmesgln_pci(*this, "NPorts: {}", n_ports);
u8 n_cc = m_cap_regs->structural_parameters.n_companion_controllers;
u8 n_pcc = m_cap_regs->structural_parameters.n_ports_per_companion_controller;
dmesgln_pci(*this, "Companion Controllers: {}", n_cc);
dmesgln_pci(*this, "Ports per Companion Controllers: {}", n_pcc);
if (n_ports > n_cc * n_pcc) {
dmesgln_pci(*this, "Warning: Not all ports of the EHCI controller are addressable via companion controllers");
dmesgln_pci(*this, " Some USB 2.0 ports might not be functional");
}
u8 EECP = m_cap_regs->capability_parameters.ehci_extended_capabilities_pointer;
if (EECP) {
SpinlockLocker locker(device_identifier().operation_lock());
auto legacy_support = bit_cast<LegacySupport>(PCI::read32_locked(device_identifier(), PCI::RegisterOffset { EECP }));
if (legacy_support.HC_BIOS_owned_semaphore)
dmesgln_pci(*this, "Warning: EHCI controller is BIOS owned");
}
// FIXME: Decide which Interrupts we want
// FIXME: Detect and switch on 64 bit support
// FIXME: Allocate and initialize Task Lists
// * Synchronous
// * Asynchronous
// * Leave space for for the actual list items
// and IO scratch space in case we cannot use the buffer from the request
// FIXME: Initialize the controller and start it
// * Setup the root hub emulation
// * Enable Software routing (CF)
// * Maybe configure port power
return {};
}
}