DragonOS/kernel/src/driver/net/virtio_net.rs
LoGin b5b571e026
修复内核的clippy检查报错 (#637)
修复内核的clippy检查报错
---------

Co-authored-by: Samuel Dai <947309196@qq.com>
Co-authored-by: Donkey Kane <109840258+xiaolin2004@users.noreply.github.com>
Co-authored-by: themildwind <107623059+themildwind@users.noreply.github.com>
Co-authored-by: GnoCiYeH <heyicong@dragonos.org>
Co-authored-by: MemoryShore <105195940+MemoryShore@users.noreply.github.com>
Co-authored-by: 曾俊 <110876916+ZZJJWarth@users.noreply.github.com>
Co-authored-by: sun5etop <146408999+sun5etop@users.noreply.github.com>
Co-authored-by: hmt <114841534+1037827920@users.noreply.github.com>
Co-authored-by: laokengwt <143977175+laokengwt@users.noreply.github.com>
Co-authored-by: TTaq <103996388+TTaq@users.noreply.github.com>
Co-authored-by: Jomo <2512364506@qq.com>
Co-authored-by: Samuel Dai <samuka007@qq.com>
Co-authored-by: sspphh <112558065+sspphh@users.noreply.github.com>
2024-03-22 23:26:39 +08:00

425 lines
13 KiB
Rust
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

use core::{
cell::UnsafeCell,
fmt::Debug,
ops::{Deref, DerefMut},
};
use alloc::{
string::String,
sync::{Arc, Weak},
};
use smoltcp::{phy, wire};
use virtio_drivers::{device::net::VirtIONet, transport::Transport};
use super::NetDriver;
use crate::{
driver::{
base::{
device::{bus::Bus, driver::Driver, Device, DeviceId, IdTable},
kobject::{KObjType, KObject, KObjectState},
},
virtio::{irq::virtio_irq_manager, virtio_impl::HalImpl, VirtIODevice},
},
exception::{irqdesc::IrqReturn, IrqNumber},
kerror, kinfo,
libs::spinlock::SpinLock,
net::{generate_iface_id, net_core::poll_ifaces_try_lock_onetime, NET_DRIVERS},
time::Instant,
};
use system_error::SystemError;
/// @brief Virtio网络设备驱动(加锁)
pub struct VirtioNICDriver<T: Transport> {
pub inner: Arc<SpinLock<VirtIONet<HalImpl, T, 2>>>,
}
impl<T: Transport> Clone for VirtioNICDriver<T> {
fn clone(&self) -> Self {
return VirtioNICDriver {
inner: self.inner.clone(),
};
}
}
/// 网卡驱动的包裹器,这是为了获取网卡驱动的可变引用而设计的。
///
/// 由于smoltcp的设计导致需要在poll的时候获取网卡驱动的可变引用
/// 同时需要在token的consume里面获取可变引用。为了避免双重加锁所以需要这个包裹器。
struct VirtioNICDriverWrapper<T: Transport>(UnsafeCell<VirtioNICDriver<T>>);
unsafe impl<T: Transport> Send for VirtioNICDriverWrapper<T> {}
unsafe impl<T: Transport> Sync for VirtioNICDriverWrapper<T> {}
impl<T: Transport> Deref for VirtioNICDriverWrapper<T> {
type Target = VirtioNICDriver<T>;
fn deref(&self) -> &Self::Target {
unsafe { &*self.0.get() }
}
}
impl<T: Transport> DerefMut for VirtioNICDriverWrapper<T> {
fn deref_mut(&mut self) -> &mut Self::Target {
unsafe { &mut *self.0.get() }
}
}
#[allow(clippy::mut_from_ref)]
impl<T: Transport> VirtioNICDriverWrapper<T> {
fn force_get_mut(&self) -> &mut VirtioNICDriver<T> {
unsafe { &mut *self.0.get() }
}
}
impl<T: Transport> Debug for VirtioNICDriver<T> {
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
f.debug_struct("VirtioNICDriver").finish()
}
}
pub struct VirtioInterface<T: Transport> {
driver: VirtioNICDriverWrapper<T>,
iface_id: usize,
iface: SpinLock<smoltcp::iface::Interface>,
name: String,
dev_id: Arc<DeviceId>,
}
impl<T: Transport> Debug for VirtioInterface<T> {
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
f.debug_struct("VirtioInterface")
.field("driver", self.driver.deref())
.field("iface_id", &self.iface_id)
.field("iface", &"smoltcp::iface::Interface")
.field("name", &self.name)
.finish()
}
}
impl<T: Transport> VirtioInterface<T> {
pub fn new(mut driver: VirtioNICDriver<T>, dev_id: Arc<DeviceId>) -> Arc<Self> {
let iface_id = generate_iface_id();
let mut iface_config = smoltcp::iface::Config::new();
// todo: 随机设定这个值。
// 参见 https://docs.rs/smoltcp/latest/smoltcp/iface/struct.Config.html#structfield.random_seed
iface_config.random_seed = 12345;
iface_config.hardware_addr = Some(wire::HardwareAddress::Ethernet(
smoltcp::wire::EthernetAddress(driver.inner.lock().mac_address()),
));
let iface = smoltcp::iface::Interface::new(iface_config, &mut driver);
let driver: VirtioNICDriverWrapper<T> = VirtioNICDriverWrapper(UnsafeCell::new(driver));
let result = Arc::new(VirtioInterface {
driver,
iface_id,
iface: SpinLock::new(iface),
name: format!("eth{}", iface_id),
dev_id,
});
return result;
}
}
impl<T: Transport + 'static> VirtIODevice for VirtioInterface<T> {
fn handle_irq(&self, _irq: IrqNumber) -> Result<IrqReturn, SystemError> {
poll_ifaces_try_lock_onetime().ok();
return Ok(IrqReturn::Handled);
}
fn dev_id(&self) -> &Arc<DeviceId> {
return &self.dev_id;
}
}
impl<T: Transport> Drop for VirtioInterface<T> {
fn drop(&mut self) {
// 从全局的网卡接口信息表中删除这个网卡的接口信息
NET_DRIVERS.write_irqsave().remove(&self.iface_id);
}
}
impl<T: 'static + Transport> VirtioNICDriver<T> {
pub fn new(driver_net: VirtIONet<HalImpl, T, 2>) -> Self {
let mut iface_config = smoltcp::iface::Config::new();
// todo: 随机设定这个值。
// 参见 https://docs.rs/smoltcp/latest/smoltcp/iface/struct.Config.html#structfield.random_seed
iface_config.random_seed = 12345;
iface_config.hardware_addr = Some(wire::HardwareAddress::Ethernet(
smoltcp::wire::EthernetAddress(driver_net.mac_address()),
));
let inner: Arc<SpinLock<VirtIONet<HalImpl, T, 2>>> = Arc::new(SpinLock::new(driver_net));
let result = VirtioNICDriver { inner };
return result;
}
}
pub struct VirtioNetToken<T: Transport> {
driver: VirtioNICDriver<T>,
rx_buffer: Option<virtio_drivers::device::net::RxBuffer>,
}
impl<T: Transport> VirtioNetToken<T> {
pub fn new(
driver: VirtioNICDriver<T>,
rx_buffer: Option<virtio_drivers::device::net::RxBuffer>,
) -> Self {
return Self { driver, rx_buffer };
}
}
impl<T: Transport> phy::Device for VirtioNICDriver<T> {
type RxToken<'a> = VirtioNetToken<T> where Self: 'a;
type TxToken<'a> = VirtioNetToken<T> where Self: 'a;
fn receive(
&mut self,
_timestamp: smoltcp::time::Instant,
) -> Option<(Self::RxToken<'_>, Self::TxToken<'_>)> {
match self.inner.lock().receive() {
Ok(buf) => Some((
VirtioNetToken::new(self.clone(), Some(buf)),
VirtioNetToken::new(self.clone(), None),
)),
Err(virtio_drivers::Error::NotReady) => None,
Err(err) => panic!("VirtIO receive failed: {}", err),
}
}
fn transmit(&mut self, _timestamp: smoltcp::time::Instant) -> Option<Self::TxToken<'_>> {
// kdebug!("VirtioNet: transmit");
if self.inner.lock_irqsave().can_send() {
// kdebug!("VirtioNet: can send");
return Some(VirtioNetToken::new(self.clone(), None));
} else {
// kdebug!("VirtioNet: can not send");
return None;
}
}
fn capabilities(&self) -> phy::DeviceCapabilities {
let mut caps = phy::DeviceCapabilities::default();
// 网卡的最大传输单元. 请与IP层的MTU进行区分。这个值应当是网卡的最大传输单元而不是IP层的MTU。
caps.max_transmission_unit = 2000;
/*
Maximum burst size, in terms of MTU.
The network device is unable to send or receive bursts large than the value returned by this function.
If None, there is no fixed limit on burst size, e.g. if network buffers are dynamically allocated.
*/
caps.max_burst_size = Some(1);
return caps;
}
}
impl<T: Transport> phy::TxToken for VirtioNetToken<T> {
fn consume<R, F>(self, len: usize, f: F) -> R
where
F: FnOnce(&mut [u8]) -> R,
{
// // 为了线程安全这里需要对VirtioNet进行加【写锁】以保证对设备的互斥访问。
let mut driver_net = self.driver.inner.lock();
let mut tx_buf = driver_net.new_tx_buffer(len);
let result = f(tx_buf.packet_mut());
driver_net.send(tx_buf).expect("virtio_net send failed");
return result;
}
}
impl<T: Transport> phy::RxToken for VirtioNetToken<T> {
fn consume<R, F>(self, f: F) -> R
where
F: FnOnce(&mut [u8]) -> R,
{
// 为了线程安全这里需要对VirtioNet进行加【写锁】以保证对设备的互斥访问。
let mut rx_buf = self.rx_buffer.unwrap();
let result = f(rx_buf.packet_mut());
self.driver
.inner
.lock()
.recycle_rx_buffer(rx_buf)
.expect("virtio_net recv failed");
result
}
}
/// @brief virtio-net 驱动的初始化与测试
pub fn virtio_net<T: Transport + 'static>(transport: T, dev_id: Arc<DeviceId>) {
let driver_net: VirtIONet<HalImpl, T, 2> =
match VirtIONet::<HalImpl, T, 2>::new(transport, 4096) {
Ok(net) => net,
Err(_) => {
kerror!("VirtIONet init failed");
return;
}
};
let mac = smoltcp::wire::EthernetAddress::from_bytes(&driver_net.mac_address());
let driver: VirtioNICDriver<T> = VirtioNICDriver::new(driver_net);
let iface = VirtioInterface::new(driver, dev_id);
let name = iface.name.clone();
// 将网卡的接口信息注册到全局的网卡接口信息表中
NET_DRIVERS
.write_irqsave()
.insert(iface.nic_id(), iface.clone());
virtio_irq_manager()
.register_device(iface.clone())
.expect("Register virtio net failed");
kinfo!(
"Virtio-net driver init successfully!\tNetDevID: [{}], MAC: [{}]",
name,
mac
);
}
impl<T: Transport + 'static> Driver for VirtioInterface<T> {
fn id_table(&self) -> Option<IdTable> {
todo!()
}
fn add_device(&self, _device: Arc<dyn Device>) {
todo!()
}
fn delete_device(&self, _device: &Arc<dyn Device>) {
todo!()
}
fn devices(&self) -> alloc::vec::Vec<Arc<dyn Device>> {
todo!()
}
fn bus(&self) -> Option<Weak<dyn Bus>> {
todo!()
}
fn set_bus(&self, _bus: Option<Weak<dyn Bus>>) {
todo!()
}
}
impl<T: Transport + 'static> NetDriver for VirtioInterface<T> {
fn mac(&self) -> smoltcp::wire::EthernetAddress {
let mac: [u8; 6] = self.driver.inner.lock().mac_address();
return smoltcp::wire::EthernetAddress::from_bytes(&mac);
}
#[inline]
fn nic_id(&self) -> usize {
return self.iface_id;
}
#[inline]
fn name(&self) -> String {
return self.name.clone();
}
fn update_ip_addrs(&self, ip_addrs: &[wire::IpCidr]) -> Result<(), SystemError> {
if ip_addrs.len() != 1 {
return Err(SystemError::EINVAL);
}
self.iface.lock().update_ip_addrs(|addrs| {
let dest = addrs.iter_mut().next();
if let Some(dest) = dest {
*dest = ip_addrs[0];
} else {
addrs.push(ip_addrs[0]).expect("Push ipCidr failed: full");
}
});
return Ok(());
}
fn poll(&self, sockets: &mut smoltcp::iface::SocketSet) -> Result<(), SystemError> {
let timestamp: smoltcp::time::Instant = Instant::now().into();
let mut guard = self.iface.lock();
let poll_res = guard.poll(timestamp, self.driver.force_get_mut(), sockets);
// todo: notify!!!
// kdebug!("Virtio Interface poll:{poll_res}");
if poll_res {
return Ok(());
}
return Err(SystemError::EAGAIN_OR_EWOULDBLOCK);
}
#[inline(always)]
fn inner_iface(&self) -> &SpinLock<smoltcp::iface::Interface> {
return &self.iface;
}
// fn as_any_ref(&'static self) -> &'static dyn core::any::Any {
// return self;
// }
}
impl<T: Transport + 'static> KObject for VirtioInterface<T> {
fn as_any_ref(&self) -> &dyn core::any::Any {
self
}
fn set_inode(&self, _inode: Option<Arc<crate::filesystem::kernfs::KernFSInode>>) {
todo!()
}
fn inode(&self) -> Option<Arc<crate::filesystem::kernfs::KernFSInode>> {
todo!()
}
fn parent(&self) -> Option<alloc::sync::Weak<dyn KObject>> {
todo!()
}
fn set_parent(&self, _parent: Option<alloc::sync::Weak<dyn KObject>>) {
todo!()
}
fn kset(&self) -> Option<Arc<crate::driver::base::kset::KSet>> {
todo!()
}
fn set_kset(&self, _kset: Option<Arc<crate::driver::base::kset::KSet>>) {
todo!()
}
fn kobj_type(&self) -> Option<&'static dyn crate::driver::base::kobject::KObjType> {
todo!()
}
fn name(&self) -> String {
self.name.clone()
}
fn set_name(&self, _name: String) {
todo!()
}
fn kobj_state(
&self,
) -> crate::libs::rwlock::RwLockReadGuard<crate::driver::base::kobject::KObjectState> {
todo!()
}
fn kobj_state_mut(
&self,
) -> crate::libs::rwlock::RwLockWriteGuard<crate::driver::base::kobject::KObjectState> {
todo!()
}
fn set_kobj_state(&self, _state: KObjectState) {
todo!()
}
fn set_kobj_type(&self, _ktype: Option<&'static dyn KObjType>) {
todo!()
}
}
// 向编译器保证VirtioNICDriver在线程之间是安全的.
// 由于smoltcp只会在token内真正操作网卡设备并且在VirtioNetToken的consume
// 方法内会对VirtioNet进行加【写锁】因此能够保证对设备操作的的互斥访问
// 因此VirtioNICDriver在线程之间是安全的。
// unsafe impl<T: Transport> Sync for VirtioNICDriver<T> {}
// unsafe impl<T: Transport> Send for VirtioNICDriver<T> {}