实现Ps/2鼠标驱动 (#500)

实现Ps/2鼠标驱动
This commit is contained in:
R0ronoa
2024-01-24 19:17:22 +08:00
committed by GitHub
parent 8d72b68da9
commit a381e482cb
9 changed files with 965 additions and 420 deletions

View File

@ -1,2 +1,3 @@
pub mod ps2_dev;
pub mod ps2_mouse;
pub mod serio;

View File

@ -0,0 +1,2 @@
pub mod ps_mouse_device;
pub mod ps_mouse_driver;

View File

@ -0,0 +1,672 @@
use core::hint::spin_loop;
use alloc::{
string::ToString,
sync::{Arc, Weak},
};
use kdepends::ringbuffer::{AllocRingBuffer, RingBuffer};
use system_error::SystemError;
use unified_init::macros::unified_init;
use crate::{
arch::{io::PortIOArch, CurrentIrqArch, CurrentPortIOArch},
driver::{
base::{
class::Class,
device::{
bus::Bus, device_manager, device_number::DeviceNumber, driver::Driver, Device,
DeviceType, IdTable,
},
kobject::{KObjType, KObject, KObjectState, LockedKObjectState},
kset::KSet,
},
input::serio::serio_device::{serio_device_manager, SerioDevice},
},
exception::InterruptArch,
filesystem::{
devfs::{devfs_register, DevFS, DeviceINode},
kernfs::KernFSInode,
vfs::{
core::generate_inode_id, syscall::ModeType, FilePrivateData, FileSystem, FileType,
IndexNode, Metadata,
},
},
init::initcall::INITCALL_DEVICE,
libs::{
rwlock::{RwLockReadGuard, RwLockWriteGuard},
spinlock::SpinLock,
},
time::TimeSpec,
};
static mut PS2_MOUSE_DEVICE: Option<Arc<Ps2MouseDevice>> = None;
pub fn ps2_mouse_device() -> Option<Arc<Ps2MouseDevice>> {
unsafe { PS2_MOUSE_DEVICE.clone() }
}
const ADDRESS_PORT_ADDRESS: u16 = 0x64;
const DATA_PORT_ADDRESS: u16 = 0x60;
const KEYBOARD_COMMAND_ENABLE_PS2_MOUSE_PORT: u8 = 0xa8;
const KEYBOARD_COMMAND_SEND_TO_PS2_MOUSE: u8 = 0xd4;
const MOUSE_BUFFER_CAPACITY: usize = 15;
bitflags! {
/// Represents the flags currently set for the mouse.
#[derive(Default)]
pub struct MouseFlags: u8 {
/// Whether or not the left mouse button is pressed.
const LEFT_BUTTON = 0b0000_0001;
/// Whether or not the right mouse button is pressed.
const RIGHT_BUTTON = 0b0000_0010;
/// Whether or not the middle mouse button is pressed.
const MIDDLE_BUTTON = 0b0000_0100;
/// Whether or not the packet is valid or not.
const ALWAYS_ONE = 0b0000_1000;
/// Whether or not the x delta is negative.
const X_SIGN = 0b0001_0000;
/// Whether or not the y delta is negative.
const Y_SIGN = 0b0010_0000;
/// Whether or not the x delta overflowed.
const X_OVERFLOW = 0b0100_0000;
/// Whether or not the y delta overflowed.
const Y_OVERFLOW = 0b1000_0000;
}
}
#[derive(Debug)]
enum PsMouseCommand {
SampleRate(u8),
EnablePacketStreaming,
// SetDefaults = 0xF6,
InitKeyboard,
GetMouseId,
SetSampleRate,
}
impl Into<u8> for PsMouseCommand {
fn into(self) -> u8 {
match self {
Self::SampleRate(x) => x,
Self::EnablePacketStreaming => 0xf4,
Self::InitKeyboard => 0x47,
Self::GetMouseId => 0xf2,
Self::SetSampleRate => 0xf3,
}
}
}
#[derive(Debug)]
pub struct MouseState {
flags: MouseFlags,
x: i16,
y: i16,
}
#[allow(dead_code)]
impl MouseState {
/// Returns a new `MouseState`.
pub const fn new() -> MouseState {
MouseState {
flags: MouseFlags::empty(),
x: 0,
y: 0,
}
}
/// Returns true if the left mouse button is currently down.
pub fn left_button_down(&self) -> bool {
self.flags.contains(MouseFlags::LEFT_BUTTON)
}
/// Returns true if the left mouse button is currently up.
pub fn left_button_up(&self) -> bool {
!self.flags.contains(MouseFlags::LEFT_BUTTON)
}
/// Returns true if the right mouse button is currently down.
pub fn right_button_down(&self) -> bool {
self.flags.contains(MouseFlags::RIGHT_BUTTON)
}
/// Returns true if the right mouse button is currently up.
pub fn right_button_up(&self) -> bool {
!self.flags.contains(MouseFlags::RIGHT_BUTTON)
}
/// Returns true if the x axis has moved.
pub fn x_moved(&self) -> bool {
self.x != 0
}
/// Returns true if the y axis has moved.
pub fn y_moved(&self) -> bool {
self.y != 0
}
/// Returns true if the x or y axis has moved.
pub fn moved(&self) -> bool {
self.x_moved() || self.y_moved()
}
/// Returns the x delta of the mouse state.
pub fn get_x(&self) -> i16 {
self.x
}
/// Returns the y delta of the mouse state.
pub fn get_y(&self) -> i16 {
self.y
}
}
#[derive(Debug)]
#[cast_to([sync] Device, SerioDevice)]
pub struct Ps2MouseDevice {
inner: SpinLock<InnerPs2MouseDevice>,
kobj_state: LockedKObjectState,
}
impl Ps2MouseDevice {
pub const NAME: &'static str = "psmouse";
pub fn new() -> Self {
let r = Self {
inner: SpinLock::new(InnerPs2MouseDevice {
bus: None,
class: None,
driver: None,
kern_inode: None,
parent: None,
kset: None,
kobj_type: None,
current_packet: 0,
current_state: MouseState::new(),
buf: AllocRingBuffer::new(MOUSE_BUFFER_CAPACITY),
devfs_metadata: Metadata {
dev_id: 1,
inode_id: generate_inode_id(),
size: 4096,
blk_size: 0,
blocks: 0,
atime: TimeSpec::default(),
mtime: TimeSpec::default(),
ctime: TimeSpec::default(),
file_type: FileType::CharDevice, // 文件夹block设备char设备
mode: ModeType::from_bits_truncate(0o644),
nlinks: 1,
uid: 0,
gid: 0,
raw_dev: DeviceNumber::default(), // 这里用来作为device number
},
device_inode_fs: None,
}),
kobj_state: LockedKObjectState::new(None),
};
return r;
}
pub fn init(&self) -> Result<(), SystemError> {
let _irq_guard = unsafe { CurrentIrqArch::save_and_disable_irq() };
self.write_control_port(KEYBOARD_COMMAND_ENABLE_PS2_MOUSE_PORT)?;
for _i in 0..1000 {
for _j in 0..1000 {
spin_loop();
}
}
self.read_data_port().ok();
self.send_command_to_ps2mouse(PsMouseCommand::EnablePacketStreaming)
.map_err(|e| {
kerror!("ps2 mouse init error: {:?}", e);
e
})?;
self.read_data_port().ok();
for _i in 0..1000 {
for _j in 0..1000 {
spin_loop();
}
}
// self.send_command_to_ps2mouse(PsMouseCommand::InitKeyboard)?;
self.do_send_command(DATA_PORT_ADDRESS as u8, PsMouseCommand::InitKeyboard.into())?;
self.read_data_port().ok();
for _i in 0..1000 {
for _j in 0..1000 {
spin_loop();
}
}
self.set_sample_rate(20)?;
// self.get_mouse_id()?;
Ok(())
}
#[allow(dead_code)]
pub fn get_mouse_id(&self) -> Result<(), SystemError> {
self.send_command_to_ps2mouse(PsMouseCommand::GetMouseId)?;
let _mouse_id = self.read_data_port()?;
Ok(())
}
/// 设置鼠标采样率
///
/// `hz` 合法值为 10,20,40,60,80,100,200
pub fn set_sample_rate(&self, hz: u8) -> Result<(), SystemError> {
const SAMPLE_RATE: [u8; 7] = [10, 20, 40, 60, 80, 100, 200];
if !SAMPLE_RATE.contains(&hz) {
return Err(SystemError::EINVAL);
}
self.send_command_to_ps2mouse(PsMouseCommand::SetSampleRate)?;
self.read_data_port().ok();
for _i in 0..1000 {
for _j in 0..1000 {
spin_loop();
}
}
self.send_command_to_ps2mouse(PsMouseCommand::SampleRate(hz))?;
for _i in 0..1000 {
for _j in 0..1000 {
spin_loop();
}
}
self.read_data_port().ok();
Ok(())
}
/// # 函数的功能
/// 鼠标设备处理数据包
pub fn process_packet(&self) -> Result<(), SystemError> {
let packet = self.read_data_port()?;
let mut guard = self.inner.lock();
guard.buf.push(packet); // 更新缓冲区
match guard.current_packet {
0 => {
let flags: MouseFlags = MouseFlags::from_bits_truncate(packet);
if !flags.contains(MouseFlags::ALWAYS_ONE) {
return Ok(());
}
guard.current_state.flags = flags;
}
1 => {
let flags = guard.current_state.flags.clone();
if !flags.contains(MouseFlags::X_OVERFLOW) {
guard.current_state.x = self.get_x_movement(packet, flags);
}
}
2 => {
let flags = guard.current_state.flags.clone();
if !flags.contains(MouseFlags::Y_OVERFLOW) {
guard.current_state.y = self.get_y_movement(packet, flags);
}
// kdebug!(
// "Ps2MouseDevice packet : flags:{}, x:{}, y:{}\n",
// guard.current_state.flags.bits,
// guard.current_state.x,
// guard.current_state.y
// )
}
_ => unreachable!(),
}
guard.current_packet = (guard.current_packet + 1) % 3;
Ok(())
}
fn get_x_movement(&self, packet: u8, flags: MouseFlags) -> i16 {
if flags.contains(MouseFlags::X_SIGN) {
return self.sign_extend(packet);
} else {
return packet as i16;
}
}
fn get_y_movement(&self, packet: u8, flags: MouseFlags) -> i16 {
if flags.contains(MouseFlags::Y_SIGN) {
return self.sign_extend(packet);
} else {
return packet as i16;
}
}
fn sign_extend(&self, packet: u8) -> i16 {
((packet as u16) | 0xFF00) as i16
}
fn read_data_port(&self) -> Result<u8, SystemError> {
self.wait_for_write()?;
let cmd = unsafe { CurrentPortIOArch::in8(ADDRESS_PORT_ADDRESS) };
if (cmd & 0x21) == 0x21 {
let data = unsafe { CurrentPortIOArch::in8(DATA_PORT_ADDRESS) };
return Ok(data);
} else {
return Err(SystemError::ENODATA);
}
}
#[inline(never)]
fn send_command_to_ps2mouse(&self, command: PsMouseCommand) -> Result<(), SystemError> {
self.do_send_command(KEYBOARD_COMMAND_SEND_TO_PS2_MOUSE, command.into())?;
Ok(())
}
#[inline(never)]
fn do_send_command(&self, ctrl: u8, command: u8) -> Result<(), SystemError> {
self.write_control_port(ctrl)?;
self.write_data_port(command)?;
return Ok(());
}
fn write_data_port(&self, data: u8) -> Result<(), SystemError> {
self.wait_for_write()?;
unsafe {
CurrentPortIOArch::out8(DATA_PORT_ADDRESS, data);
}
Ok(())
}
fn write_control_port(&self, command: u8) -> Result<(), SystemError> {
self.wait_for_write()?;
unsafe {
CurrentPortIOArch::out8(ADDRESS_PORT_ADDRESS, command);
}
Ok(())
}
fn wait_for_read(&self) -> Result<(), SystemError> {
let timeout = 100_000;
for _ in 0..timeout {
let value = unsafe { CurrentPortIOArch::in8(ADDRESS_PORT_ADDRESS) };
if (value & 0x1) == 0x1 {
return Ok(());
}
}
Err(SystemError::ETIMEDOUT)
}
fn wait_for_write(&self) -> Result<(), SystemError> {
let timeout = 100_000;
for _ in 0..timeout {
let value = unsafe { CurrentPortIOArch::in8(ADDRESS_PORT_ADDRESS) };
if (value & 0x2) == 0 {
return Ok(());
}
}
Err(SystemError::ETIMEDOUT)
}
}
#[derive(Debug)]
struct InnerPs2MouseDevice {
bus: Option<Weak<dyn Bus>>,
class: Option<Arc<dyn Class>>,
driver: Option<Weak<dyn Driver>>,
kern_inode: Option<Arc<KernFSInode>>,
parent: Option<Weak<dyn KObject>>,
kset: Option<Arc<KSet>>,
kobj_type: Option<&'static dyn KObjType>,
/// 鼠标数据
current_state: MouseState,
current_packet: u8,
/// 鼠标数据环形缓冲区
buf: AllocRingBuffer<u8>,
/// device inode要求的字段
device_inode_fs: Option<Weak<DevFS>>,
devfs_metadata: Metadata,
}
impl Device for Ps2MouseDevice {
fn is_dead(&self) -> bool {
false
}
fn dev_type(&self) -> DeviceType {
DeviceType::Char
}
fn id_table(&self) -> IdTable {
IdTable::new(self.name().to_string(), None)
}
fn set_bus(&self, bus: Option<alloc::sync::Weak<dyn Bus>>) {
self.inner.lock_irqsave().bus = bus;
}
fn set_class(&self, class: Option<alloc::sync::Arc<dyn Class>>) {
self.inner.lock_irqsave().class = class;
}
fn driver(&self) -> Option<alloc::sync::Arc<dyn Driver>> {
self.inner.lock_irqsave().driver.clone()?.upgrade()
}
fn set_driver(&self, driver: Option<alloc::sync::Weak<dyn Driver>>) {
self.inner.lock_irqsave().driver = driver;
}
fn can_match(&self) -> bool {
true
}
fn set_can_match(&self, _can_match: bool) {}
fn state_synced(&self) -> bool {
true
}
fn bus(&self) -> Option<alloc::sync::Weak<dyn Bus>> {
self.inner.lock_irqsave().bus.clone()
}
fn class(&self) -> Option<Arc<dyn Class>> {
self.inner.lock_irqsave().class.clone()
}
}
impl SerioDevice for Ps2MouseDevice {
fn write(
&self,
_device: &alloc::sync::Arc<dyn SerioDevice>,
_data: u8,
) -> Result<(), system_error::SystemError> {
todo!()
}
fn open(
&self,
_device: &alloc::sync::Arc<dyn SerioDevice>,
) -> Result<(), system_error::SystemError> {
todo!()
}
fn close(
&self,
_device: &alloc::sync::Arc<dyn SerioDevice>,
) -> Result<(), system_error::SystemError> {
todo!()
}
fn start(
&self,
_device: &alloc::sync::Arc<dyn SerioDevice>,
) -> Result<(), system_error::SystemError> {
todo!()
}
fn stop(
&self,
_device: &alloc::sync::Arc<dyn SerioDevice>,
) -> Result<(), system_error::SystemError> {
todo!()
}
}
impl KObject for Ps2MouseDevice {
fn as_any_ref(&self) -> &dyn core::any::Any {
self
}
fn set_inode(&self, inode: Option<alloc::sync::Arc<KernFSInode>>) {
self.inner.lock_irqsave().kern_inode = inode;
}
fn inode(&self) -> Option<alloc::sync::Arc<KernFSInode>> {
self.inner.lock_irqsave().kern_inode.clone()
}
fn parent(&self) -> Option<alloc::sync::Weak<dyn KObject>> {
self.inner.lock_irqsave().parent.clone()
}
fn set_parent(&self, parent: Option<alloc::sync::Weak<dyn KObject>>) {
self.inner.lock_irqsave().parent = parent
}
fn kset(&self) -> Option<alloc::sync::Arc<KSet>> {
self.inner.lock_irqsave().kset.clone()
}
fn set_kset(&self, kset: Option<alloc::sync::Arc<KSet>>) {
self.inner.lock_irqsave().kset = kset;
}
fn kobj_type(&self) -> Option<&'static dyn KObjType> {
self.inner.lock_irqsave().kobj_type.clone()
}
fn set_kobj_type(&self, ktype: Option<&'static dyn KObjType>) {
self.inner.lock_irqsave().kobj_type = ktype;
}
fn name(&self) -> alloc::string::String {
Self::NAME.to_string()
}
fn set_name(&self, _name: alloc::string::String) {}
fn kobj_state(&self) -> RwLockReadGuard<KObjectState> {
self.kobj_state.read()
}
fn kobj_state_mut(&self) -> RwLockWriteGuard<KObjectState> {
self.kobj_state.write()
}
fn set_kobj_state(&self, state: KObjectState) {
*self.kobj_state.write() = state;
}
}
impl DeviceINode for Ps2MouseDevice {
fn set_fs(&self, fs: Weak<DevFS>) {
self.inner.lock_irqsave().device_inode_fs = Some(fs);
}
}
impl IndexNode for Ps2MouseDevice {
fn open(
&self,
_data: &mut FilePrivateData,
_mode: &crate::filesystem::vfs::file::FileMode,
) -> Result<(), SystemError> {
let mut guard = self.inner.lock_irqsave();
guard.buf.clear();
Ok(())
}
fn close(&self, _data: &mut FilePrivateData) -> Result<(), SystemError> {
let mut guard = self.inner.lock_irqsave();
guard.buf.clear();
Ok(())
}
fn read_at(
&self,
_offset: usize,
_len: usize,
buf: &mut [u8],
_data: &mut FilePrivateData,
) -> Result<usize, SystemError> {
let mut guard = self.inner.lock_irqsave();
if guard.buf.len() >= 3 {
for i in 0..3 {
buf[i] = guard.buf.dequeue().unwrap();
}
return Ok(3);
} else {
return Ok(0);
}
}
fn write_at(
&self,
_offset: usize,
_len: usize,
_buf: &[u8],
_data: &mut FilePrivateData,
) -> Result<usize, SystemError> {
return Err(SystemError::EOPNOTSUPP_OR_ENOTSUP);
}
fn fs(&self) -> Arc<dyn FileSystem> {
self.inner
.lock_irqsave()
.device_inode_fs
.as_ref()
.unwrap()
.upgrade()
.unwrap()
}
fn as_any_ref(&self) -> &dyn core::any::Any {
self
}
fn list(&self) -> Result<alloc::vec::Vec<alloc::string::String>, SystemError> {
todo!()
}
fn metadata(&self) -> Result<Metadata, SystemError> {
Ok(self.inner.lock_irqsave().devfs_metadata.clone())
}
fn resize(&self, _len: usize) -> Result<(), SystemError> {
Ok(())
}
}
#[unified_init(INITCALL_DEVICE)]
fn rs_ps2_mouse_device_int() -> Result<(), SystemError> {
kdebug!("ps2_mouse_device initializing...");
let psmouse = Arc::new(Ps2MouseDevice::new());
device_manager().device_default_initialize(&(psmouse.clone() as Arc<dyn Device>));
serio_device_manager().register_port(psmouse.clone())?;
devfs_register(&psmouse.name(), psmouse.clone()).map_err(|e| {
kerror!(
"register psmouse device '{}' to devfs failed: {:?}",
psmouse.name(),
e
);
device_manager().remove(&(psmouse.clone() as Arc<dyn Device>));
e
})?;
unsafe { PS2_MOUSE_DEVICE = Some(psmouse) };
return Ok(());
}

View File

@ -0,0 +1,260 @@
use alloc::{
string::ToString,
sync::{Arc, Weak},
vec::Vec,
};
use system_error::SystemError;
use unified_init::macros::unified_init;
use crate::{
arch::{io::PortIOArch, CurrentPortIOArch},
driver::{
base::{
device::{bus::Bus, driver::Driver, Device, IdTable},
kobject::{KObjType, KObject, KObjectState, LockedKObjectState},
kset::KSet,
},
input::serio::{
serio_device::SerioDevice,
serio_driver::{serio_driver_manager, SerioDriver},
},
},
filesystem::kernfs::KernFSInode,
init::initcall::INITCALL_DEVICE,
libs::{
rwlock::{RwLockReadGuard, RwLockWriteGuard},
spinlock::SpinLock,
},
};
use super::ps_mouse_device::{ps2_mouse_device, Ps2MouseDevice};
extern "C" {
fn c_ps2_mouse_init();
}
#[no_mangle]
unsafe extern "C" fn ps2_mouse_driver_interrupt() {
if let Some(psmouse_device) = ps2_mouse_device() {
ps2_mouse_driver()
.interrupt(&(psmouse_device as Arc<dyn SerioDevice>), 0, 0)
.ok();
} else {
unsafe { CurrentPortIOArch::in8(0x60) };
}
}
static mut PS2_MOUSE_DRIVER: Option<Arc<Ps2MouseDriver>> = None;
#[allow(dead_code)]
pub fn ps2_mouse_driver() -> Arc<Ps2MouseDriver> {
unsafe { PS2_MOUSE_DRIVER.clone().unwrap() }
}
#[derive(Debug)]
#[cast_to([sync] Driver)]
#[cast_to([sync] SerioDriver)]
pub struct Ps2MouseDriver {
inner: SpinLock<InnerPs2MouseDriver>,
kobj_state: LockedKObjectState,
}
impl Ps2MouseDriver {
pub const NAME: &'static str = "psmouse";
pub fn new() -> Arc<Self> {
let r = Arc::new(Ps2MouseDriver {
inner: SpinLock::new(InnerPs2MouseDriver {
ktype: None,
kset: None,
parent: None,
kernfs_inode: None,
devices: Vec::new(),
bus: None,
self_ref: Weak::new(),
}),
kobj_state: LockedKObjectState::new(None),
});
r.inner.lock().self_ref = Arc::downgrade(&r);
return r;
}
#[allow(dead_code)]
pub fn process_packet(&self) {
let guard = self.inner.lock();
if guard.devices.is_empty() {
return;
}
let device: Option<&Ps2MouseDevice> = guard.devices[0]
.as_any_ref()
.downcast_ref::<Ps2MouseDevice>();
let _ = device.unwrap().process_packet();
}
}
#[derive(Debug)]
pub struct InnerPs2MouseDriver {
ktype: Option<&'static dyn KObjType>,
kset: Option<Arc<KSet>>,
parent: Option<Weak<dyn KObject>>,
kernfs_inode: Option<Arc<KernFSInode>>,
devices: Vec<Arc<dyn Device>>,
bus: Option<Weak<dyn Bus>>,
self_ref: Weak<Ps2MouseDriver>,
}
impl Driver for Ps2MouseDriver {
fn id_table(&self) -> Option<IdTable> {
Some(IdTable::new("psmouse".to_string(), None))
}
fn devices(&self) -> alloc::vec::Vec<Arc<dyn Device>> {
self.inner.lock().devices.clone()
}
fn add_device(&self, device: Arc<dyn Device>) {
let mut guard = self.inner.lock();
// check if the device is already in the list
if guard.devices.iter().any(|dev| Arc::ptr_eq(dev, &device)) {
return;
}
guard.devices.push(device);
}
fn delete_device(&self, device: &Arc<dyn Device>) {
let mut guard = self.inner.lock();
guard.devices.retain(|dev| !Arc::ptr_eq(dev, device));
}
fn set_bus(&self, bus: Option<alloc::sync::Weak<dyn Bus>>) {
self.inner.lock().bus = bus;
}
fn bus(&self) -> Option<Weak<dyn Bus>> {
self.inner.lock().bus.clone()
}
}
impl KObject for Ps2MouseDriver {
fn as_any_ref(&self) -> &dyn core::any::Any {
self
}
fn set_inode(&self, inode: Option<Arc<KernFSInode>>) {
self.inner.lock().kernfs_inode = inode;
}
fn inode(&self) -> Option<Arc<KernFSInode>> {
self.inner.lock().kernfs_inode.clone()
}
fn parent(&self) -> Option<alloc::sync::Weak<dyn KObject>> {
self.inner.lock().parent.clone()
}
fn set_parent(&self, parent: Option<alloc::sync::Weak<dyn KObject>>) {
self.inner.lock().parent = parent;
}
fn kset(&self) -> Option<Arc<KSet>> {
self.inner.lock().kset.clone()
}
fn set_kset(&self, kset: Option<Arc<KSet>>) {
self.inner.lock().kset = kset;
}
fn kobj_type(&self) -> Option<&'static dyn KObjType> {
self.inner.lock().ktype
}
fn set_kobj_type(&self, ktype: Option<&'static dyn KObjType>) {
self.inner.lock().ktype = ktype;
}
fn name(&self) -> alloc::string::String {
Self::NAME.to_string()
}
fn set_name(&self, _name: alloc::string::String) {}
fn kobj_state(&self) -> RwLockReadGuard<KObjectState> {
self.kobj_state.read()
}
fn kobj_state_mut(&self) -> RwLockWriteGuard<KObjectState> {
self.kobj_state.write()
}
fn set_kobj_state(&self, state: KObjectState) {
*self.kobj_state.write() = state;
}
}
impl SerioDriver for Ps2MouseDriver {
fn write_wakeup(
&self,
_device: &Arc<dyn SerioDevice>,
) -> Result<(), system_error::SystemError> {
todo!()
}
fn interrupt(
&self,
device: &Arc<dyn SerioDevice>,
_char: u8,
_int: u8,
) -> Result<(), system_error::SystemError> {
let device = device
.clone()
.arc_any()
.downcast::<Ps2MouseDevice>()
.map_err(|_| SystemError::EINVAL)?;
device.process_packet()?;
Ok(())
}
fn connect(&self, device: &Arc<dyn SerioDevice>) -> Result<(), system_error::SystemError> {
let device = device
.clone()
.arc_any()
.downcast::<Ps2MouseDevice>()
.map_err(|_| SystemError::EINVAL)?;
device.set_driver(Some(self.inner.lock_irqsave().self_ref.clone()));
device.init()?;
unsafe { c_ps2_mouse_init() };
return Ok(());
}
fn reconnect(&self, _device: &Arc<dyn SerioDevice>) -> Result<(), system_error::SystemError> {
todo!()
}
fn fast_reconnect(
&self,
_device: &Arc<dyn SerioDevice>,
) -> Result<(), system_error::SystemError> {
todo!()
}
fn disconnect(&self, _device: &Arc<dyn SerioDevice>) -> Result<(), system_error::SystemError> {
todo!()
}
fn cleanup(&self, _device: &Arc<dyn SerioDevice>) -> Result<(), system_error::SystemError> {
todo!()
}
}
#[unified_init(INITCALL_DEVICE)]
fn ps2_mouse_driver_init() -> Result<(), SystemError> {
kdebug!("Ps2_mouse_drive initing...");
let driver = Ps2MouseDriver::new();
serio_driver_manager().register(driver.clone())?;
unsafe { PS2_MOUSE_DRIVER = Some(driver) };
return Ok(());
}

View File

@ -18,8 +18,7 @@ pub fn serio_bus() -> Arc<SerioBus> {
}
/// # 函数的功能
///
/// 初始化serio bus
/// 初始化Serio总线
///
/// 参考: https://code.dragonos.org.cn/xref/linux-6.1.9/drivers/input/serio/serio.c#1024
pub fn serio_bus_init() -> Result<(), SystemError> {

View File

@ -1,7 +1,7 @@
use alloc::sync::Arc;
use system_error::SystemError;
use crate::driver::base::device::{bus::Bus, Device};
use crate::driver::base::device::{bus::Bus, device_manager, Device};
use super::serio_bus;
@ -31,7 +31,6 @@ pub trait SerioDevice: Device {
fn stop(&self, device: &Arc<dyn SerioDevice>) -> Result<(), SystemError>;
}
#[allow(dead_code)]
#[inline(always)]
pub fn serio_device_manager() -> &'static SerioDeviceManager {
&SerioDeviceManager
@ -39,7 +38,6 @@ pub fn serio_device_manager() -> &'static SerioDeviceManager {
pub struct SerioDeviceManager;
#[allow(dead_code)]
impl SerioDeviceManager {
/// # 函数功能
/// 注册Serio设备
@ -65,6 +63,7 @@ impl SerioDeviceManager {
/// todohttps://code.dragonos.org.cn/xref/linux-6.1.9/drivers/input/serio/serio.c#494
pub fn init_port(&self, device: Arc<dyn SerioDevice>) -> Result<(), SystemError> {
device.set_bus(Some(Arc::downgrade(&(serio_bus() as Arc<dyn Bus>))));
device_manager().add_device(device.clone() as Arc<dyn Device>)?;
Ok(())
}
}

View File

@ -24,6 +24,8 @@ pub trait SerioDriver: Driver {
///
/// ## 返回值
/// 无
///
/// todo:https://code.dragonos.org.cn/xref/linux-6.1.9/drivers/input/serio/serio.c?fi=__serio_register_driver#989
fn interrupt(
&self,
device: &Arc<dyn SerioDevice>,
@ -43,6 +45,12 @@ pub trait SerioDriver: Driver {
}
///todo: https://code.dragonos.org.cn/xref/linux-6.1.9/drivers/input/serio/serio.c#810
#[inline(always)]
pub fn serio_driver_manager() -> &'static SerioDriverManager {
&SerioDriverManager
}
pub struct SerioDriverManager;
#[allow(dead_code)]