ax-driver 0.6.0

ArceOS rdrive driver registration and rdif binding collection
Documentation
extern crate alloc;

use alloc::boxed::Box;

use rd_net::{Interface, NetError};
use rdrive::{Device, DriverGeneric, probe::pci::EndpointRc};

pub struct PlatformNetDevice {
    name: &'static str,
    net: Option<rd_net::Net>,
    irq_num: Option<usize>,
}

impl PlatformNetDevice {
    fn new(name: &'static str, net: rd_net::Net, irq_num: Option<usize>) -> Self {
        Self {
            name,
            net: Some(net),
            irq_num,
        }
    }

    pub fn take_net(&mut self) -> Option<(rd_net::Net, &'static str, Option<usize>)> {
        Some((self.net.take()?, self.name, self.irq_num))
    }
}

pub fn take_rd_net_device(
    device: Device<PlatformNetDevice>,
) -> Result<(rd_net::Net, &'static str, Option<usize>), NetError> {
    let mut dev = device
        .lock()
        .map_err(|_| NetError::Other(Box::new(rd_net::KError::Unknown("device locked"))))?;
    dev.take_net()
        .ok_or_else(|| NetError::Other(Box::new(rd_net::KError::Unknown("device already taken"))))
}

impl DriverGeneric for PlatformNetDevice {
    fn name(&self) -> &str {
        self.name
    }
}

pub fn pci_legacy_irq(endpoint: &EndpointRc) -> Option<usize> {
    #[cfg(all(feature = "pci-fdt", target_os = "none"))]
    {
        let interrupt_pin = endpoint.interrupt_pin();
        if interrupt_pin != 0 {
            match crate::pci::fdt_irq_for_endpoint(endpoint.address(), interrupt_pin) {
                Ok(Some(irq)) => return Some(irq),
                Ok(None) => {}
                Err(err) => log::warn!(
                    "failed to resolve FDT IRQ for net endpoint {}: {err}",
                    endpoint.address()
                ),
            }
        }
    }

    #[cfg(feature = "pci")]
    if let Some(irq) =
        crate::pci::legacy_irq_for_endpoint(endpoint.address(), endpoint.interrupt_pin())
    {
        return Some(irq);
    }

    let line = endpoint.interrupt_line();
    if line == 0 || line == u8::MAX {
        return None;
    }
    Some(pci_legacy_line_to_irq(line))
}

const fn pci_legacy_line_to_irq(line: u8) -> usize {
    const PCI_IRQ_BASE: usize = if cfg!(target_arch = "x86_64") || cfg!(target_arch = "riscv64") {
        if cfg!(target_arch = "x86_64") {
            0x20
        } else {
            0
        }
    } else {
        0
    };

    PCI_IRQ_BASE + line as usize
}

pub trait PlatformDeviceNet {
    fn register_net<T>(self, name: &'static str, dev: T, irq_num: Option<usize>)
    where
        T: Interface + 'static;
}

impl PlatformDeviceNet for rdrive::PlatformDevice {
    fn register_net<T>(self, name: &'static str, dev: T, irq_num: Option<usize>)
    where
        T: Interface + 'static,
    {
        let net = rd_net::Net::new(dev, axklib::dma::op());
        self.register(PlatformNetDevice::new(name, net, irq_num));
    }
}