refact: embed crate mouce, uinput (#8836)
Signed-off-by: fufesou <linlong1266@gmail.com>
This commit is contained in:
		
							parent
							
								
									541d9c6b86
								
							
						
					
					
						commit
						c0e9445602
					
				
							
								
								
									
										9
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										9
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							| @ -3777,14 +3777,6 @@ dependencies = [ | ||||
|  "windows-sys 0.48.0", | ||||
| ] | ||||
| 
 | ||||
| [[package]] | ||||
| name = "mouce" | ||||
| version = "0.2.1" | ||||
| source = "git+https://github.com/rustdesk-org/mouce.git#177625a395cd8fa73964714d0039535cb9b47893" | ||||
| dependencies = [ | ||||
|  "glob", | ||||
| ] | ||||
| 
 | ||||
| [[package]] | ||||
| name = "muda" | ||||
| version = "0.13.5" | ||||
| @ -5514,7 +5506,6 @@ dependencies = [ | ||||
|  "libpulse-simple-binding", | ||||
|  "mac_address", | ||||
|  "magnum-opus", | ||||
|  "mouce", | ||||
|  "num_cpus", | ||||
|  "objc", | ||||
|  "objc_id", | ||||
|  | ||||
| @ -152,7 +152,6 @@ psimple = { package = "libpulse-simple-binding", version = "2.27" } | ||||
| pulse = { package = "libpulse-binding", version = "2.27" } | ||||
| rust-pulsectl = { git = "https://github.com/open-trade/pulsectl" } | ||||
| async-process = "1.7" | ||||
| mouce = { git="https://github.com/rustdesk-org/mouce.git" } | ||||
| evdev = { git="https://github.com/rustdesk-org/evdev" } | ||||
| dbus = "0.9" | ||||
| dbus-crossroads = "0.5" | ||||
|  | ||||
| @ -4,7 +4,11 @@ use evdev::{ | ||||
|     uinput::{VirtualDevice, VirtualDeviceBuilder}, | ||||
|     AttributeSet, EventType, InputEvent, | ||||
| }; | ||||
| use hbb_common::{allow_err, bail, log, tokio::{self, runtime::Runtime}, ResultType}; | ||||
| use hbb_common::{ | ||||
|     allow_err, bail, log, | ||||
|     tokio::{self, runtime::Runtime}, | ||||
|     ResultType, | ||||
| }; | ||||
| 
 | ||||
| static IPC_CONN_TIMEOUT: u64 = 1000; | ||||
| static IPC_REQUEST_TIMEOUT: u64 = 1000; | ||||
| @ -34,7 +38,10 @@ pub mod client { | ||||
|         fn send_get_key_state(&mut self, data: Data) -> ResultType<bool> { | ||||
|             self.rt.block_on(self.conn.send(&data))?; | ||||
| 
 | ||||
|             match self.rt.block_on(self.conn.next_timeout(IPC_REQUEST_TIMEOUT)) { | ||||
|             match self | ||||
|                 .rt | ||||
|                 .block_on(self.conn.next_timeout(IPC_REQUEST_TIMEOUT)) | ||||
|             { | ||||
|                 Ok(Some(Data::KeyboardResponse(ipc::DataKeyboardResponse::GetKeyState(state)))) => { | ||||
|                     Ok(state) | ||||
|                 } | ||||
| @ -171,7 +178,6 @@ pub mod client { | ||||
| pub mod service { | ||||
|     use super::*; | ||||
|     use hbb_common::lazy_static; | ||||
|     use mouce::MouseActions; | ||||
|     use std::{collections::HashMap, sync::Mutex}; | ||||
| 
 | ||||
|     lazy_static::lazy_static! { | ||||
| @ -431,7 +437,8 @@ pub mod service { | ||||
|             DataKeyboard::KeyDown(key) => { | ||||
|                 if let Ok((k, is_shift)) = map_key(key) { | ||||
|                     if is_shift { | ||||
|                         let down_event = InputEvent::new(EventType::KEY, evdev::Key::KEY_LEFTSHIFT.code(), 1); | ||||
|                         let down_event = | ||||
|                             InputEvent::new(EventType::KEY, evdev::Key::KEY_LEFTSHIFT.code(), 1); | ||||
|                         allow_err!(keyboard.emit(&[down_event])); | ||||
|                     } | ||||
|                     let down_event = InputEvent::new(EventType::KEY, k.code(), 1); | ||||
| @ -504,7 +511,7 @@ pub mod service { | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fn handle_mouse(mouse: &mut mouce::nix::UInputMouseManager, data: &DataMouse) { | ||||
|     fn handle_mouse(mouse: &mut mouce::UInputMouseManager, data: &DataMouse) { | ||||
|         log::trace!("handle_mouse {:?}", &data); | ||||
|         match data { | ||||
|             DataMouse::MoveTo(x, y) => { | ||||
| @ -515,9 +522,9 @@ pub mod service { | ||||
|             } | ||||
|             DataMouse::Down(button) => { | ||||
|                 let btn = match button { | ||||
|                     enigo::MouseButton::Left => mouce::common::MouseButton::Left, | ||||
|                     enigo::MouseButton::Middle => mouce::common::MouseButton::Middle, | ||||
|                     enigo::MouseButton::Right => mouce::common::MouseButton::Right, | ||||
|                     enigo::MouseButton::Left => mouce::MouseButton::Left, | ||||
|                     enigo::MouseButton::Middle => mouce::MouseButton::Middle, | ||||
|                     enigo::MouseButton::Right => mouce::MouseButton::Right, | ||||
|                     _ => { | ||||
|                         return; | ||||
|                     } | ||||
| @ -526,9 +533,9 @@ pub mod service { | ||||
|             } | ||||
|             DataMouse::Up(button) => { | ||||
|                 let btn = match button { | ||||
|                     enigo::MouseButton::Left => mouce::common::MouseButton::Left, | ||||
|                     enigo::MouseButton::Middle => mouce::common::MouseButton::Middle, | ||||
|                     enigo::MouseButton::Right => mouce::common::MouseButton::Right, | ||||
|                     enigo::MouseButton::Left => mouce::MouseButton::Left, | ||||
|                     enigo::MouseButton::Middle => mouce::MouseButton::Middle, | ||||
|                     enigo::MouseButton::Right => mouce::MouseButton::Right, | ||||
|                     _ => { | ||||
|                         return; | ||||
|                     } | ||||
| @ -537,9 +544,9 @@ pub mod service { | ||||
|             } | ||||
|             DataMouse::Click(button) => { | ||||
|                 let btn = match button { | ||||
|                     enigo::MouseButton::Left => mouce::common::MouseButton::Left, | ||||
|                     enigo::MouseButton::Middle => mouce::common::MouseButton::Middle, | ||||
|                     enigo::MouseButton::Right => mouce::common::MouseButton::Right, | ||||
|                     enigo::MouseButton::Left => mouce::MouseButton::Left, | ||||
|                     enigo::MouseButton::Middle => mouce::MouseButton::Middle, | ||||
|                     enigo::MouseButton::Right => mouce::MouseButton::Right, | ||||
|                     _ => { | ||||
|                         return; | ||||
|                     } | ||||
| @ -553,9 +560,9 @@ pub mod service { | ||||
|                 let mut length = *length; | ||||
| 
 | ||||
|                 let scroll = if length < 0 { | ||||
|                     mouce::common::ScrollDirection::Up | ||||
|                     mouce::ScrollDirection::Up | ||||
|                 } else { | ||||
|                     mouce::common::ScrollDirection::Down | ||||
|                     mouce::ScrollDirection::Down | ||||
|                 }; | ||||
| 
 | ||||
|                 if length < 0 { | ||||
| @ -621,7 +628,7 @@ pub mod service { | ||||
|                 rng_y.0, | ||||
|                 rng_y.1 | ||||
|             ); | ||||
|             let mut mouse = match mouce::Mouse::new_uinput(rng_x, rng_y) { | ||||
|             let mut mouse = match mouce::UInputMouseManager::new(rng_x, rng_y) { | ||||
|                 Ok(mouse) => mouse, | ||||
|                 Err(e) => { | ||||
|                     log::error!("Failed to create mouse, {}", e); | ||||
| @ -650,7 +657,7 @@ pub mod service { | ||||
|                                                 rng_y.0, | ||||
|                                                 rng_y.1 | ||||
|                                             ); | ||||
|                                             mouse = match mouce::Mouse::new_uinput(rng_x, rng_y) { | ||||
|                                             mouse = match mouce::UInputMouseManager::new(rng_x, rng_y) { | ||||
|                                                 Ok(mouse) => mouse, | ||||
|                                                 Err(e) => { | ||||
|                                                     log::error!("Failed to create mouse, {}", e); | ||||
| @ -761,3 +768,346 @@ pub mod service { | ||||
|         log::info!("stop uinput control service"); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| // https://github.com/emrebicer/mouce
 | ||||
| mod mouce { | ||||
|     use std::{ | ||||
|         fs::File, | ||||
|         io::{Error, ErrorKind, Result}, | ||||
|         mem::size_of, | ||||
|         os::{ | ||||
|             raw::{c_char, c_int, c_long, c_uint, c_ulong, c_ushort}, | ||||
|             unix::{fs::OpenOptionsExt, io::AsRawFd}, | ||||
|         }, | ||||
|         thread, | ||||
|         time::Duration, | ||||
|     }; | ||||
| 
 | ||||
|     pub const O_NONBLOCK: c_int = 2048; | ||||
| 
 | ||||
|     /// ioctl and uinput definitions
 | ||||
|     const UI_ABS_SETUP: c_ulong = 1075598596; | ||||
|     const UI_SET_EVBIT: c_ulong = 1074025828; | ||||
|     const UI_SET_KEYBIT: c_ulong = 1074025829; | ||||
|     const UI_SET_RELBIT: c_ulong = 1074025830; | ||||
|     const UI_SET_ABSBIT: c_ulong = 1074025831; | ||||
|     const UI_DEV_SETUP: c_ulong = 1079792899; | ||||
|     const UI_DEV_CREATE: c_ulong = 21761; | ||||
|     const UI_DEV_DESTROY: c_uint = 21762; | ||||
| 
 | ||||
|     pub const EV_KEY: c_int = 0x01; | ||||
|     pub const EV_REL: c_int = 0x02; | ||||
|     pub const EV_ABS: c_int = 0x03; | ||||
|     pub const REL_X: c_uint = 0x00; | ||||
|     pub const REL_Y: c_uint = 0x01; | ||||
|     pub const ABS_X: c_uint = 0x00; | ||||
|     pub const ABS_Y: c_uint = 0x01; | ||||
|     pub const REL_WHEEL: c_uint = 0x08; | ||||
|     pub const REL_HWHEEL: c_uint = 0x06; | ||||
|     pub const BTN_LEFT: c_int = 0x110; | ||||
|     pub const BTN_RIGHT: c_int = 0x111; | ||||
|     pub const BTN_MIDDLE: c_int = 0x112; | ||||
|     pub const BTN_SIDE: c_int = 0x113; | ||||
|     pub const BTN_EXTRA: c_int = 0x114; | ||||
|     pub const BTN_FORWARD: c_int = 0x115; | ||||
|     pub const BTN_BACK: c_int = 0x116; | ||||
|     pub const BTN_TASK: c_int = 0x117; | ||||
|     const SYN_REPORT: c_int = 0x00; | ||||
|     const EV_SYN: c_int = 0x00; | ||||
|     const BUS_USB: c_ushort = 0x03; | ||||
| 
 | ||||
|     /// uinput types
 | ||||
|     #[repr(C)] | ||||
|     struct UInputSetup { | ||||
|         id: InputId, | ||||
|         name: [c_char; UINPUT_MAX_NAME_SIZE], | ||||
|         ff_effects_max: c_ulong, | ||||
|     } | ||||
| 
 | ||||
|     #[repr(C)] | ||||
|     struct InputId { | ||||
|         bustype: c_ushort, | ||||
|         vendor: c_ushort, | ||||
|         product: c_ushort, | ||||
|         version: c_ushort, | ||||
|     } | ||||
| 
 | ||||
|     #[repr(C)] | ||||
|     pub struct InputEvent { | ||||
|         pub time: TimeVal, | ||||
|         pub r#type: c_ushort, | ||||
|         pub code: c_ushort, | ||||
|         pub value: c_int, | ||||
|     } | ||||
| 
 | ||||
|     #[repr(C)] | ||||
|     pub struct TimeVal { | ||||
|         pub tv_sec: c_ulong, | ||||
|         pub tv_usec: c_ulong, | ||||
|     } | ||||
| 
 | ||||
|     #[repr(C)] | ||||
|     pub struct UinputAbsSetup { | ||||
|         pub code: c_ushort, | ||||
|         pub absinfo: InputAbsinfo, | ||||
|     } | ||||
| 
 | ||||
|     #[repr(C)] | ||||
|     pub struct InputAbsinfo { | ||||
|         pub value: c_int, | ||||
|         pub minimum: c_int, | ||||
|         pub maximum: c_int, | ||||
|         pub fuzz: c_int, | ||||
|         pub flat: c_int, | ||||
|         pub resolution: c_int, | ||||
|     } | ||||
| 
 | ||||
|     extern "C" { | ||||
|         fn ioctl(fd: c_int, request: c_ulong, ...) -> c_int; | ||||
|         fn write(fd: c_int, buf: *mut InputEvent, count: usize) -> c_long; | ||||
|     } | ||||
| 
 | ||||
|     #[derive(Debug, Copy, Clone)] | ||||
|     pub enum MouseButton { | ||||
|         Left, | ||||
|         Middle, | ||||
|         Side, | ||||
|         Extra, | ||||
|         Right, | ||||
|         Back, | ||||
|         Forward, | ||||
|         Task, | ||||
|     } | ||||
| 
 | ||||
|     #[derive(Debug, Copy, Clone)] | ||||
|     pub enum ScrollDirection { | ||||
|         Up, | ||||
|         Down, | ||||
|         Right, | ||||
|         Left, | ||||
|     } | ||||
| 
 | ||||
|     const UINPUT_MAX_NAME_SIZE: usize = 80; | ||||
| 
 | ||||
|     pub struct UInputMouseManager { | ||||
|         uinput_file: File, | ||||
|     } | ||||
| 
 | ||||
|     impl UInputMouseManager { | ||||
|         pub fn new(rng_x: (i32, i32), rng_y: (i32, i32)) -> Result<Self> { | ||||
|             let manager = UInputMouseManager { | ||||
|                 uinput_file: File::options() | ||||
|                     .write(true) | ||||
|                     .custom_flags(O_NONBLOCK) | ||||
|                     .open("/dev/uinput")?, | ||||
|             }; | ||||
|             let fd = manager.uinput_file.as_raw_fd(); | ||||
|             unsafe { | ||||
|                 // For press events (also needed for mouse movement)
 | ||||
|                 ioctl(fd, UI_SET_EVBIT, EV_KEY); | ||||
|                 ioctl(fd, UI_SET_KEYBIT, BTN_LEFT); | ||||
|                 ioctl(fd, UI_SET_KEYBIT, BTN_RIGHT); | ||||
|                 ioctl(fd, UI_SET_KEYBIT, BTN_MIDDLE); | ||||
| 
 | ||||
|                 // For mouse movement
 | ||||
|                 ioctl(fd, UI_SET_EVBIT, EV_ABS); | ||||
|                 ioctl(fd, UI_SET_ABSBIT, ABS_X); | ||||
|                 ioctl( | ||||
|                     fd, | ||||
|                     UI_ABS_SETUP, | ||||
|                     &UinputAbsSetup { | ||||
|                         code: ABS_X as _, | ||||
|                         absinfo: InputAbsinfo { | ||||
|                             value: 0, | ||||
|                             minimum: rng_x.0, | ||||
|                             maximum: rng_x.1, | ||||
|                             fuzz: 0, | ||||
|                             flat: 0, | ||||
|                             resolution: 0, | ||||
|                         }, | ||||
|                     }, | ||||
|                 ); | ||||
|                 ioctl(fd, UI_SET_ABSBIT, ABS_Y); | ||||
|                 ioctl( | ||||
|                     fd, | ||||
|                     UI_ABS_SETUP, | ||||
|                     &UinputAbsSetup { | ||||
|                         code: ABS_Y as _, | ||||
|                         absinfo: InputAbsinfo { | ||||
|                             value: 0, | ||||
|                             minimum: rng_y.0, | ||||
|                             maximum: rng_y.1, | ||||
|                             fuzz: 0, | ||||
|                             flat: 0, | ||||
|                             resolution: 0, | ||||
|                         }, | ||||
|                     }, | ||||
|                 ); | ||||
| 
 | ||||
|                 ioctl(fd, UI_SET_EVBIT, EV_REL); | ||||
|                 ioctl(fd, UI_SET_RELBIT, REL_X); | ||||
|                 ioctl(fd, UI_SET_RELBIT, REL_Y); | ||||
|                 ioctl(fd, UI_SET_RELBIT, REL_WHEEL); | ||||
|                 ioctl(fd, UI_SET_RELBIT, REL_HWHEEL); | ||||
|             } | ||||
| 
 | ||||
|             let mut usetup = UInputSetup { | ||||
|                 id: InputId { | ||||
|                     bustype: BUS_USB, | ||||
|                     // Random vendor and product
 | ||||
|                     vendor: 0x2222, | ||||
|                     product: 0x3333, | ||||
|                     version: 0, | ||||
|                 }, | ||||
|                 name: [0; UINPUT_MAX_NAME_SIZE], | ||||
|                 ff_effects_max: 0, | ||||
|             }; | ||||
| 
 | ||||
|             let mut device_bytes: Vec<c_char> = "mouce-library-fake-mouse" | ||||
|                 .chars() | ||||
|                 .map(|ch| ch as c_char) | ||||
|                 .collect(); | ||||
| 
 | ||||
|             // Fill the rest of the name buffer with empty chars
 | ||||
|             for _ in 0..UINPUT_MAX_NAME_SIZE - device_bytes.len() { | ||||
|                 device_bytes.push('\0' as c_char); | ||||
|             } | ||||
| 
 | ||||
|             usetup.name.copy_from_slice(&device_bytes); | ||||
| 
 | ||||
|             unsafe { | ||||
|                 ioctl(fd, UI_DEV_SETUP, &usetup); | ||||
|                 ioctl(fd, UI_DEV_CREATE); | ||||
|             } | ||||
| 
 | ||||
|             // On UI_DEV_CREATE the kernel will create the device node for this
 | ||||
|             // device. We are inserting a pause here so that userspace has time
 | ||||
|             // to detect, initialize the new device, and can start listening to
 | ||||
|             // the event, otherwise it will not notice the event we are about to send.
 | ||||
|             thread::sleep(Duration::from_millis(300)); | ||||
| 
 | ||||
|             Ok(manager) | ||||
|         } | ||||
| 
 | ||||
|         /// Write the given event to the uinput file
 | ||||
|         fn emit(&self, r#type: c_int, code: c_int, value: c_int) -> Result<()> { | ||||
|             let mut event = InputEvent { | ||||
|                 time: TimeVal { | ||||
|                     tv_sec: 0, | ||||
|                     tv_usec: 0, | ||||
|                 }, | ||||
|                 r#type: r#type as c_ushort, | ||||
|                 code: code as c_ushort, | ||||
|                 value, | ||||
|             }; | ||||
|             let fd = self.uinput_file.as_raw_fd(); | ||||
| 
 | ||||
|             unsafe { | ||||
|                 let count = size_of::<InputEvent>(); | ||||
|                 let written_bytes = write(fd, &mut event, count); | ||||
|                 if written_bytes == -1 || written_bytes != count as c_long { | ||||
|                     return Err(Error::new( | ||||
|                         ErrorKind::Other, | ||||
|                         format!("failed while trying to write to a file"), | ||||
|                     )); | ||||
|                 } | ||||
|             } | ||||
| 
 | ||||
|             Ok(()) | ||||
|         } | ||||
| 
 | ||||
|         /// Syncronize the device
 | ||||
|         fn syncronize(&self) -> Result<()> { | ||||
|             self.emit(EV_SYN, SYN_REPORT, 0)?; | ||||
|             // Give uinput some time to update the mouse location,
 | ||||
|             // otherwise it fails to move the mouse on release mode
 | ||||
|             // A delay of 1 milliseconds seems to be enough for it
 | ||||
|             thread::sleep(Duration::from_millis(1)); | ||||
|             Ok(()) | ||||
|         } | ||||
| 
 | ||||
|         /// Move the mouse relative to the current position
 | ||||
|         fn move_relative_(&self, x: i32, y: i32) -> Result<()> { | ||||
|             // uinput does not move the mouse in pixels but uses `units`. I couldn't
 | ||||
|             // find information regarding to this uinput `unit`, but according to
 | ||||
|             // my findings 1 unit corresponds to exactly 2 pixels.
 | ||||
|             //
 | ||||
|             // To achieve the expected behavior; divide the parameters by 2
 | ||||
|             //
 | ||||
|             // This seems like there is a bug in this crate, but the
 | ||||
|             // behavior is the same on other projects that make use of
 | ||||
|             // uinput. e.g. `ydotool`. When you try to move your mouse,
 | ||||
|             // it will move 2x further pixels
 | ||||
|             self.emit(EV_REL, REL_X as c_int, (x as f32 / 2.).ceil() as c_int)?; | ||||
|             self.emit(EV_REL, REL_Y as c_int, (y as f32 / 2.).ceil() as c_int)?; | ||||
|             self.syncronize() | ||||
|         } | ||||
| 
 | ||||
|         fn map_btn(button: &MouseButton) -> c_int { | ||||
|             match button { | ||||
|                 MouseButton::Left => BTN_LEFT, | ||||
|                 MouseButton::Right => BTN_RIGHT, | ||||
|                 MouseButton::Middle => BTN_MIDDLE, | ||||
|                 MouseButton::Side => BTN_SIDE, | ||||
|                 MouseButton::Extra => BTN_EXTRA, | ||||
|                 MouseButton::Forward => BTN_FORWARD, | ||||
|                 MouseButton::Back => BTN_BACK, | ||||
|                 MouseButton::Task => BTN_TASK, | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         pub fn move_to(&self, x: usize, y: usize) -> Result<()> { | ||||
|             // // For some reason, absolute mouse move events are not working on uinput
 | ||||
|             // // (as I understand those events are intended for touch events)
 | ||||
|             // //
 | ||||
|             // // As a work around solution; first set the mouse to top left, then
 | ||||
|             // // call relative move function to simulate an absolute move event
 | ||||
|             //self.move_relative(i32::MIN, i32::MIN)?;
 | ||||
|             //self.move_relative(x as i32, y as i32)
 | ||||
| 
 | ||||
|             self.emit(EV_ABS, ABS_X as c_int, x as c_int)?; | ||||
|             self.emit(EV_ABS, ABS_Y as c_int, y as c_int)?; | ||||
|             self.syncronize() | ||||
|         } | ||||
| 
 | ||||
|         pub fn move_relative(&self, x_offset: i32, y_offset: i32) -> Result<()> { | ||||
|             self.move_relative_(x_offset, y_offset) | ||||
|         } | ||||
| 
 | ||||
|         pub fn press_button(&self, button: &MouseButton) -> Result<()> { | ||||
|             self.emit(EV_KEY, Self::map_btn(button), 1)?; | ||||
|             self.syncronize() | ||||
|         } | ||||
| 
 | ||||
|         pub fn release_button(&self, button: &MouseButton) -> Result<()> { | ||||
|             self.emit(EV_KEY, Self::map_btn(button), 0)?; | ||||
|             self.syncronize() | ||||
|         } | ||||
| 
 | ||||
|         pub fn click_button(&self, button: &MouseButton) -> Result<()> { | ||||
|             self.press_button(button)?; | ||||
|             self.release_button(button) | ||||
|         } | ||||
| 
 | ||||
|         pub fn scroll_wheel(&self, direction: &ScrollDirection) -> Result<()> { | ||||
|             let (code, scroll_value) = match direction { | ||||
|                 ScrollDirection::Up => (REL_WHEEL, 1), | ||||
|                 ScrollDirection::Down => (REL_WHEEL, -1), | ||||
|                 ScrollDirection::Left => (REL_HWHEEL, -1), | ||||
|                 ScrollDirection::Right => (REL_HWHEEL, 1), | ||||
|             }; | ||||
|             self.emit(EV_REL, code as c_int, scroll_value)?; | ||||
|             self.syncronize() | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     impl Drop for UInputMouseManager { | ||||
|         fn drop(&mut self) { | ||||
|             let fd = self.uinput_file.as_raw_fd(); | ||||
|             unsafe { | ||||
|                 // Destroy the device, the file is closed automatically by the File module
 | ||||
|                 ioctl(fd, UI_DEV_DESTROY as c_ulong); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user