// SPDX-License-Identifier: GPL-2.0 // Copyright (C) 2025 Google LLC. //! Traits for rendering or updating values exported to DebugFS. use crate::prelude::*; use crate::sync::Mutex; use crate::uaccess::UserSliceReader; use core::fmt::{self, Debug, Formatter}; use core::str::FromStr; use core::sync::atomic::{ AtomicI16, AtomicI32, AtomicI64, AtomicI8, AtomicIsize, AtomicU16, AtomicU32, AtomicU64, AtomicU8, AtomicUsize, Ordering, }; /// 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 Formatter<'_>) -> fmt::Result; } impl Writer for Mutex { fn write(&self, f: &mut Formatter<'_>) -> fmt::Result { self.lock().write(f) } } impl Writer for T { fn write(&self, f: &mut Formatter<'_>) -> fmt::Result { writeln!(f, "{self:?}") } } /// 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(()) } } macro_rules! impl_reader_for_atomic { ($(($atomic_type:ty, $int_type:ty)),*) => { $( impl Reader for $atomic_type { 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::<$int_type>().map_err(|_| EINVAL)?; self.store(val, Ordering::Relaxed); Ok(()) } } )* }; } impl_reader_for_atomic!( (AtomicI16, i16), (AtomicI32, i32), (AtomicI64, i64), (AtomicI8, i8), (AtomicIsize, isize), (AtomicU16, u16), (AtomicU32, u32), (AtomicU64, u64), (AtomicU8, u8), (AtomicUsize, usize) );