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));
}
}