// SPDX-License-Identifier: GPL-2.0 // Copyright (C) 2025 Google LLC. //! Traits for rendering or updating values exported to DebugFS. use crate::alloc::Allocator; use crate::fmt; use crate::fs::file; use crate::prelude::*; use crate::sync::atomic::{Atomic, AtomicBasicOps, AtomicType, Relaxed}; use crate::sync::Arc; use crate::sync::Mutex; use crate::transmute::{AsBytes, FromBytes}; use crate::uaccess::{UserSliceReader, UserSliceWriter}; use core::ops::{Deref, DerefMut}; use core::str::FromStr; /// A trait for types that can be written into a string. /// /// This works very similarly to `Debug`, and is automatically implemented if `Debug` is /// implemented for a type. It is also implemented for any writable type inside a `Mutex`. /// /// The derived implementation of `Debug` [may /// change](https://doc.rust-lang.org/std/fmt/trait.Debug.html#stability) /// between Rust versions, so if stability is key for your use case, please implement `Writer` /// explicitly instead. pub trait Writer { /// Formats the value using the given formatter. fn write(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result; } impl Writer for Mutex { fn write(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { self.lock().write(f) } } impl Writer for T { fn write(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { writeln!(f, "{self:?}") } } /// Trait for types that can be written out as binary. pub trait BinaryWriter { /// Writes the binary form of `self` into `writer`. /// /// `offset` is the requested offset into the binary representation of `self`. /// /// On success, returns the number of bytes written in to `writer`. fn write_to_slice( &self, writer: &mut UserSliceWriter, offset: &mut file::Offset, ) -> Result; } // Base implementation for any `T: AsBytes`. impl BinaryWriter for T { fn write_to_slice( &self, writer: &mut UserSliceWriter, offset: &mut file::Offset, ) -> Result { writer.write_slice_file(self.as_bytes(), offset) } } // Delegate for `Mutex`: Support a `T` with an outer mutex. impl BinaryWriter for Mutex { fn write_to_slice( &self, writer: &mut UserSliceWriter, offset: &mut file::Offset, ) -> Result { let guard = self.lock(); guard.write_to_slice(writer, offset) } } // Delegate for `Box`: Support a `Box` with no lock or an inner lock. impl BinaryWriter for Box where T: BinaryWriter, A: Allocator, { fn write_to_slice( &self, writer: &mut UserSliceWriter, offset: &mut file::Offset, ) -> Result { self.deref().write_to_slice(writer, offset) } } // Delegate for `Pin>`: Support a `Pin>` with no lock or an inner lock. impl BinaryWriter for Pin> where T: BinaryWriter, A: Allocator, { fn write_to_slice( &self, writer: &mut UserSliceWriter, offset: &mut file::Offset, ) -> Result { self.deref().write_to_slice(writer, offset) } } // Delegate for `Arc`: Support a `Arc` with no lock or an inner lock. impl BinaryWriter for Arc where T: BinaryWriter, { fn write_to_slice( &self, writer: &mut UserSliceWriter, offset: &mut file::Offset, ) -> Result { self.deref().write_to_slice(writer, offset) } } // Delegate for `Vec`. impl BinaryWriter for Vec where T: AsBytes, A: Allocator, { fn write_to_slice( &self, writer: &mut UserSliceWriter, offset: &mut file::Offset, ) -> Result { let slice = self.as_slice(); // SAFETY: `T: AsBytes` allows us to treat `&[T]` as `&[u8]`. let buffer = unsafe { core::slice::from_raw_parts(slice.as_ptr().cast(), core::mem::size_of_val(slice)) }; writer.write_slice_file(buffer, offset) } } /// A trait for types that can be updated from a user slice. /// /// This works similarly to `FromStr`, but operates on a `UserSliceReader` rather than a &str. /// /// It is automatically implemented for all atomic integers, or any type that implements `FromStr` /// wrapped in a `Mutex`. pub trait Reader { /// Updates the value from the given user slice. fn read_from_slice(&self, reader: &mut UserSliceReader) -> Result; } impl Reader for Mutex { fn read_from_slice(&self, reader: &mut UserSliceReader) -> Result { let mut buf = [0u8; 128]; if reader.len() > buf.len() { return Err(EINVAL); } let n = reader.len(); reader.read_slice(&mut buf[..n])?; let s = core::str::from_utf8(&buf[..n]).map_err(|_| EINVAL)?; let val = s.trim().parse::().map_err(|_| EINVAL)?; *self.lock() = val; Ok(()) } } impl Reader for Atomic where T::Repr: AtomicBasicOps, { fn read_from_slice(&self, reader: &mut UserSliceReader) -> Result { let mut buf = [0u8; 21]; // Enough for a 64-bit number. if reader.len() > buf.len() { return Err(EINVAL); } let n = reader.len(); reader.read_slice(&mut buf[..n])?; let s = core::str::from_utf8(&buf[..n]).map_err(|_| EINVAL)?; let val = s.trim().parse::().map_err(|_| EINVAL)?; self.store(val, Relaxed); Ok(()) } } /// Trait for types that can be constructed from a binary representation. /// /// See also [`BinaryReader`] for interior mutability. pub trait BinaryReaderMut { /// Reads the binary form of `self` from `reader`. /// /// Same as [`BinaryReader::read_from_slice`], but takes a mutable reference. /// /// `offset` is the requested offset into the binary representation of `self`. /// /// On success, returns the number of bytes read from `reader`. fn read_from_slice_mut( &mut self, reader: &mut UserSliceReader, offset: &mut file::Offset, ) -> Result; } // Base implementation for any `T: AsBytes + FromBytes`. impl BinaryReaderMut for T { fn read_from_slice_mut( &mut self, reader: &mut UserSliceReader, offset: &mut file::Offset, ) -> Result { reader.read_slice_file(self.as_bytes_mut(), offset) } } // Delegate for `Box`: Support a `Box` with an outer lock. impl BinaryReaderMut for Box { fn read_from_slice_mut( &mut self, reader: &mut UserSliceReader, offset: &mut file::Offset, ) -> Result { self.deref_mut().read_from_slice_mut(reader, offset) } } // Delegate for `Vec`: Support a `Vec` with an outer lock. impl BinaryReaderMut for Vec where T: AsBytes + FromBytes, A: Allocator, { fn read_from_slice_mut( &mut self, reader: &mut UserSliceReader, offset: &mut file::Offset, ) -> Result { let slice = self.as_mut_slice(); // SAFETY: `T: AsBytes + FromBytes` allows us to treat `&mut [T]` as `&mut [u8]`. let buffer = unsafe { core::slice::from_raw_parts_mut( slice.as_mut_ptr().cast(), core::mem::size_of_val(slice), ) }; reader.read_slice_file(buffer, offset) } } /// Trait for types that can be constructed from a binary representation. /// /// See also [`BinaryReaderMut`] for the mutable version. pub trait BinaryReader { /// Reads the binary form of `self` from `reader`. /// /// `offset` is the requested offset into the binary representation of `self`. /// /// On success, returns the number of bytes read from `reader`. fn read_from_slice( &self, reader: &mut UserSliceReader, offset: &mut file::Offset, ) -> Result; } // Delegate for `Mutex`: Support a `T` with an outer `Mutex`. impl BinaryReader for Mutex { fn read_from_slice( &self, reader: &mut UserSliceReader, offset: &mut file::Offset, ) -> Result { let mut this = self.lock(); this.read_from_slice_mut(reader, offset) } } // Delegate for `Box`: Support a `Box` with an inner lock. impl BinaryReader for Box { fn read_from_slice( &self, reader: &mut UserSliceReader, offset: &mut file::Offset, ) -> Result { self.deref().read_from_slice(reader, offset) } } // Delegate for `Pin>`: Support a `Pin>` with an inner lock. impl BinaryReader for Pin> { fn read_from_slice( &self, reader: &mut UserSliceReader, offset: &mut file::Offset, ) -> Result { self.deref().read_from_slice(reader, offset) } } // Delegate for `Arc`: Support an `Arc` with an inner lock. impl BinaryReader for Arc { fn read_from_slice( &self, reader: &mut UserSliceReader, offset: &mut file::Offset, ) -> Result { self.deref().read_from_slice(reader, offset) } }