| //! Implements typical patterns for `ioctl` usage. |
| |
| use super::{Ioctl, IoctlOutput, Opcode, RawOpcode}; |
| |
| use crate::backend::c; |
| use crate::io::Result; |
| |
| use core::marker::PhantomData; |
| use core::{fmt, mem}; |
| |
| /// Implements an `ioctl` with no real arguments. |
| pub struct NoArg<Opcode> { |
| /// The opcode. |
| _opcode: PhantomData<Opcode>, |
| } |
| |
| impl<Opcode: CompileTimeOpcode> fmt::Debug for NoArg<Opcode> { |
| fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { |
| f.debug_tuple("NoArg").field(&Opcode::OPCODE).finish() |
| } |
| } |
| |
| impl<Opcode: CompileTimeOpcode> NoArg<Opcode> { |
| /// Create a new no-argument `ioctl` object. |
| /// |
| /// # Safety |
| /// |
| /// - `Opcode` must provide a valid opcode. |
| #[inline] |
| pub unsafe fn new() -> Self { |
| Self { |
| _opcode: PhantomData, |
| } |
| } |
| } |
| |
| unsafe impl<Opcode: CompileTimeOpcode> Ioctl for NoArg<Opcode> { |
| type Output = (); |
| |
| const IS_MUTATING: bool = false; |
| const OPCODE: self::Opcode = Opcode::OPCODE; |
| |
| fn as_ptr(&mut self) -> *mut c::c_void { |
| core::ptr::null_mut() |
| } |
| |
| unsafe fn output_from_ptr(_: IoctlOutput, _: *mut c::c_void) -> Result<Self::Output> { |
| Ok(()) |
| } |
| } |
| |
| /// Implements the traditional "getter" pattern for `ioctl`s. |
| /// |
| /// Some `ioctl`s just read data into the userspace. As this is a popular |
| /// pattern this structure implements it. |
| pub struct Getter<Opcode, Output> { |
| /// The output data. |
| output: mem::MaybeUninit<Output>, |
| |
| /// The opcode. |
| _opcode: PhantomData<Opcode>, |
| } |
| |
| impl<Opcode: CompileTimeOpcode, Output> fmt::Debug for Getter<Opcode, Output> { |
| fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { |
| f.debug_tuple("Getter").field(&Opcode::OPCODE).finish() |
| } |
| } |
| |
| impl<Opcode: CompileTimeOpcode, Output> Getter<Opcode, Output> { |
| /// Create a new getter-style `ioctl` object. |
| /// |
| /// # Safety |
| /// |
| /// - `Opcode` must provide a valid opcode. |
| /// - For this opcode, `Output` must be the type that the kernel expects to |
| /// write into. |
| #[inline] |
| pub unsafe fn new() -> Self { |
| Self { |
| output: mem::MaybeUninit::uninit(), |
| _opcode: PhantomData, |
| } |
| } |
| } |
| |
| unsafe impl<Opcode: CompileTimeOpcode, Output> Ioctl for Getter<Opcode, Output> { |
| type Output = Output; |
| |
| const IS_MUTATING: bool = true; |
| const OPCODE: self::Opcode = Opcode::OPCODE; |
| |
| fn as_ptr(&mut self) -> *mut c::c_void { |
| self.output.as_mut_ptr().cast() |
| } |
| |
| unsafe fn output_from_ptr(_: IoctlOutput, ptr: *mut c::c_void) -> Result<Self::Output> { |
| Ok(ptr.cast::<Output>().read()) |
| } |
| } |
| |
| /// Implements the pattern for `ioctl`s where a pointer argument is given to |
| /// the `ioctl`. |
| /// |
| /// The opcode must be read-only. |
| pub struct Setter<Opcode, Input> { |
| /// The input data. |
| input: Input, |
| |
| /// The opcode. |
| _opcode: PhantomData<Opcode>, |
| } |
| |
| impl<Opcode: CompileTimeOpcode, Input: fmt::Debug> fmt::Debug for Setter<Opcode, Input> { |
| fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { |
| f.debug_tuple("Setter") |
| .field(&Opcode::OPCODE) |
| .field(&self.input) |
| .finish() |
| } |
| } |
| |
| impl<Opcode: CompileTimeOpcode, Input> Setter<Opcode, Input> { |
| /// Create a new pointer setter-style `ioctl` object. |
| /// |
| /// # Safety |
| /// |
| /// - `Opcode` must provide a valid opcode. |
| /// - For this opcode, `Input` must be the type that the kernel expects to |
| /// get. |
| #[inline] |
| pub unsafe fn new(input: Input) -> Self { |
| Self { |
| input, |
| _opcode: PhantomData, |
| } |
| } |
| } |
| |
| unsafe impl<Opcode: CompileTimeOpcode, Input> Ioctl for Setter<Opcode, Input> { |
| type Output = (); |
| |
| const IS_MUTATING: bool = false; |
| const OPCODE: self::Opcode = Opcode::OPCODE; |
| |
| fn as_ptr(&mut self) -> *mut c::c_void { |
| &mut self.input as *mut Input as *mut c::c_void |
| } |
| |
| unsafe fn output_from_ptr(_: IoctlOutput, _: *mut c::c_void) -> Result<Self::Output> { |
| Ok(()) |
| } |
| } |
| |
| /// Trait for something that provides an `ioctl` opcode at compile time. |
| pub trait CompileTimeOpcode { |
| /// The opcode. |
| const OPCODE: Opcode; |
| } |
| |
| /// Provides a bad opcode at compile time. |
| pub struct BadOpcode<const OPCODE: RawOpcode>; |
| |
| impl<const OPCODE: RawOpcode> CompileTimeOpcode for BadOpcode<OPCODE> { |
| const OPCODE: Opcode = Opcode::old(OPCODE); |
| } |
| |
| /// Provides a read code at compile time. |
| #[cfg(any(linux_kernel, bsd))] |
| pub struct ReadOpcode<const GROUP: u8, const NUM: u8, Data>(Data); |
| |
| #[cfg(any(linux_kernel, bsd))] |
| impl<const GROUP: u8, const NUM: u8, Data> CompileTimeOpcode for ReadOpcode<GROUP, NUM, Data> { |
| const OPCODE: Opcode = Opcode::read::<Data>(GROUP, NUM); |
| } |
| |
| /// Provides a write code at compile time. |
| #[cfg(any(linux_kernel, bsd))] |
| pub struct WriteOpcode<const GROUP: u8, const NUM: u8, Data>(Data); |
| |
| #[cfg(any(linux_kernel, bsd))] |
| impl<const GROUP: u8, const NUM: u8, Data> CompileTimeOpcode for WriteOpcode<GROUP, NUM, Data> { |
| const OPCODE: Opcode = Opcode::write::<Data>(GROUP, NUM); |
| } |
| |
| /// Provides a read/write code at compile time. |
| #[cfg(any(linux_kernel, bsd))] |
| pub struct ReadWriteOpcode<const GROUP: u8, const NUM: u8, Data>(Data); |
| |
| #[cfg(any(linux_kernel, bsd))] |
| impl<const GROUP: u8, const NUM: u8, Data> CompileTimeOpcode for ReadWriteOpcode<GROUP, NUM, Data> { |
| const OPCODE: Opcode = Opcode::read_write::<Data>(GROUP, NUM); |
| } |
| |
| /// Provides a `None` code at compile time. |
| #[cfg(any(linux_kernel, bsd))] |
| pub struct NoneOpcode<const GROUP: u8, const NUM: u8, Data>(Data); |
| |
| #[cfg(any(linux_kernel, bsd))] |
| impl<const GROUP: u8, const NUM: u8, Data> CompileTimeOpcode for NoneOpcode<GROUP, NUM, Data> { |
| const OPCODE: Opcode = Opcode::none::<Data>(GROUP, NUM); |
| } |