use core::{ fmt::Debug, sync::atomic::{AtomicBool, Ordering}, }; use alloc::{ string::String, sync::{Arc, Weak}, vec::Vec, }; use crate::{ libs::{ notifier::AtomicNotifierChain, rwlock::{RwLock, RwLockReadGuard}, spinlock::SpinLock, }, syscall::SystemError, }; use super::{ class::Class, device::{ bus::{Bus, BusNotifyEvent}, driver::Driver, Device, }, kset::KSet, }; /// 一个用于存储bus/class的驱动核心部分的信息的结构体 #[derive(Debug)] pub struct SubSysPrivate { /// 用于定义这个子系统的kset subsys: Arc, ksets: RwLock, /// 指向拥有当前结构体的`dyn bus`对象的弱引用 bus: SpinLock>>, /// 指向拥有当前结构体的`dyn class`对象的弱引用 class: SpinLock>>, drivers_autoprobe: AtomicBool, /// 当前总线上的所有设备 devices: RwLock>>, /// 当前总线上的所有驱动 drivers: RwLock>>, interfaces: &'static [&'static dyn SubSysInterface], bus_notifier: AtomicNotifierChain>, } #[derive(Debug)] struct SubSysKSets { /// 子系统的`devices`目录 devices_kset: Option>, /// 子系统的`drivers`目录 drivers_kset: Option>, } impl SubSysKSets { pub fn new() -> Self { return Self { devices_kset: None, drivers_kset: None, }; } } impl SubSysPrivate { pub fn new( name: String, bus: Option>, class: Option>, interfaces: &'static [&'static dyn SubSysInterface], ) -> Self { let subsys = KSet::new(name); return Self { subsys, ksets: RwLock::new(SubSysKSets::new()), drivers_autoprobe: AtomicBool::new(false), bus: SpinLock::new(bus), class: SpinLock::new(class), devices: RwLock::new(Vec::new()), drivers: RwLock::new(Vec::new()), interfaces, bus_notifier: AtomicNotifierChain::new(), }; } pub fn subsys(&self) -> Arc { return self.subsys.clone(); } #[inline] #[allow(dead_code)] pub fn bus(&self) -> Option> { return self.bus.lock().clone(); } pub fn set_bus(&self, bus: Option>) { *self.bus.lock() = bus; } #[allow(dead_code)] #[inline] pub fn class(&self) -> Option> { return self.class.lock().clone(); } pub fn set_class(&self, class: Option>) { *self.class.lock() = class; } pub fn devices(&self) -> RwLockReadGuard>> { return self.devices.read(); } pub fn drivers(&self) -> RwLockReadGuard>> { return self.drivers.read(); } pub fn drivers_autoprobe(&self) -> bool { return self.drivers_autoprobe.load(Ordering::SeqCst); } pub fn set_drivers_autoprobe(&self, drivers_autoprobe: bool) { self.drivers_autoprobe .store(drivers_autoprobe, Ordering::SeqCst); } #[allow(dead_code)] #[inline] pub fn devices_kset(&self) -> Option> { return self.ksets.read().devices_kset.clone(); } #[allow(dead_code)] #[inline] pub fn set_devices_kset(&self, devices_kset: Arc) { self.ksets.write().devices_kset = Some(devices_kset); } #[allow(dead_code)] #[inline] pub fn drivers_kset(&self) -> Option> { return self.ksets.read().drivers_kset.clone(); } pub fn set_drivers_kset(&self, drivers_kset: Arc) { self.ksets.write().drivers_kset = Some(drivers_kset); } pub fn bus_notifier(&self) -> &AtomicNotifierChain> { return &self.bus_notifier; } pub fn interfaces(&self) -> &'static [&'static dyn SubSysInterface] { return self.interfaces; } pub fn add_driver_to_vec(&self, driver: &Arc) -> Result<(), SystemError> { let mut drivers = self.drivers.write(); let driver_weak = Arc::downgrade(driver); if drivers.iter().any(|d| d.ptr_eq(&driver_weak)) { return Err(SystemError::EEXIST); } drivers.push(driver_weak); return Ok(()); } pub fn remove_driver_from_vec(&self, driver: &Arc) { let mut drivers = self.drivers.write(); let driver_weak = Arc::downgrade(driver); let index = drivers.iter().position(|d| d.ptr_eq(&driver_weak)); if let Some(index) = index { drivers.remove(index); } } pub fn add_device_to_vec(&self, device: &Arc) -> Result<(), SystemError> { let mut devices = self.devices.write(); let device_weak = Arc::downgrade(device); if devices.iter().any(|d| d.ptr_eq(&device_weak)) { return Err(SystemError::EEXIST); } devices.push(device_weak); return Ok(()); } #[allow(dead_code)] pub fn remove_device_from_vec(&self, device: &Arc) { let mut devices = self.devices.write(); let device_weak = Arc::downgrade(device); let index = devices.iter().position(|d| d.ptr_eq(&device_weak)); if let Some(index) = index { devices.remove(index); } } } /// 参考: https://opengrok.ringotek.cn/xref/linux-6.1.9/include/linux/device.h#63 pub trait SubSysInterface: Debug + Send + Sync { fn name(&self) -> &str; fn bus(&self) -> Option>; fn set_bus(&self, bus: Option>); fn add_device(&self, _device: &Arc) -> Result<(), SystemError> { return Err(SystemError::EOPNOTSUPP_OR_ENOTSUP); } fn remove_device(&self, device: &Arc); }