// SPDX-License-Identifier: GPL-2.0 // Copyright (C) 2025 Google LLC. use crate::debugfs::file_ops::FileOps; use crate::ffi::c_void; use crate::str::CStr; use crate::sync::Arc; use core::marker::PhantomData; /// Owning handle to a DebugFS entry. /// /// # Invariants /// /// The wrapped pointer will always be `NULL`, an error, or an owned DebugFS `dentry`. pub(crate) struct Entry<'a> { entry: *mut bindings::dentry, // If we were created with an owning parent, this is the keep-alive _parent: Option>>, // If we were created with a non-owning parent, this prevents us from outliving it _phantom: PhantomData<&'a ()>, } // SAFETY: [`Entry`] is just a `dentry` under the hood, which the API promises can be transferred // between threads. unsafe impl Send for Entry<'_> {} // SAFETY: All the C functions we call on the `dentry` pointer are threadsafe. unsafe impl Sync for Entry<'_> {} impl Entry<'static> { pub(crate) fn dynamic_dir(name: &CStr, parent: Option>) -> Self { let parent_ptr = match &parent { Some(entry) => entry.as_ptr(), None => core::ptr::null_mut(), }; // SAFETY: The invariants of this function's arguments ensure the safety of this call. // * `name` is a valid C string by the invariants of `&CStr`. // * `parent_ptr` is either `NULL` (if `parent` is `None`), or a pointer to a valid // `dentry` by our invariant. `debugfs_create_dir` handles `NULL` pointers correctly. let entry = unsafe { bindings::debugfs_create_dir(name.as_char_ptr(), parent_ptr) }; Entry { entry, _parent: parent, _phantom: PhantomData, } } /// # Safety /// /// * `data` must outlive the returned `Entry`. pub(crate) unsafe fn dynamic_file( name: &CStr, parent: Arc, data: &T, file_ops: &'static FileOps, ) -> Self { // SAFETY: The invariants of this function's arguments ensure the safety of this call. // * `name` is a valid C string by the invariants of `&CStr`. // * `parent.as_ptr()` is a pointer to a valid `dentry` by invariant. // * The caller guarantees that `data` will outlive the returned `Entry`. // * The guarantees on `FileOps` assert the vtable will be compatible with the data we have // provided. let entry = unsafe { bindings::debugfs_create_file_full( name.as_char_ptr(), file_ops.mode(), parent.as_ptr(), core::ptr::from_ref(data) as *mut c_void, core::ptr::null(), &**file_ops, ) }; Entry { entry, _parent: Some(parent), _phantom: PhantomData, } } } impl<'a> Entry<'a> { pub(crate) fn dir(name: &CStr, parent: Option<&'a Entry<'_>>) -> Self { let parent_ptr = match &parent { Some(entry) => entry.as_ptr(), None => core::ptr::null_mut(), }; // SAFETY: The invariants of this function's arguments ensure the safety of this call. // * `name` is a valid C string by the invariants of `&CStr`. // * `parent_ptr` is either `NULL` (if `parent` is `None`), or a pointer to a valid // `dentry` (because `parent` is a valid reference to an `Entry`). The lifetime `'a` // ensures that the parent outlives this entry. let entry = unsafe { bindings::debugfs_create_dir(name.as_char_ptr(), parent_ptr) }; Entry { entry, _parent: None, _phantom: PhantomData, } } pub(crate) fn file( name: &CStr, parent: &'a Entry<'_>, data: &'a T, file_ops: &FileOps, ) -> Self { // SAFETY: The invariants of this function's arguments ensure the safety of this call. // * `name` is a valid C string by the invariants of `&CStr`. // * `parent.as_ptr()` is a pointer to a valid `dentry` because we have `&'a Entry`. // * `data` is a valid pointer to `T` for lifetime `'a`. // * The returned `Entry` has lifetime `'a`, so it cannot outlive `parent` or `data`. // * The caller guarantees that `vtable` is compatible with `data`. // * The guarantees on `FileOps` assert the vtable will be compatible with the data we have // provided. let entry = unsafe { bindings::debugfs_create_file_full( name.as_char_ptr(), file_ops.mode(), parent.as_ptr(), core::ptr::from_ref(data) as *mut c_void, core::ptr::null(), &**file_ops, ) }; Entry { entry, _parent: None, _phantom: PhantomData, } } } impl Entry<'_> { /// Constructs a placeholder DebugFS [`Entry`]. pub(crate) fn empty() -> Self { Self { entry: core::ptr::null_mut(), _parent: None, _phantom: PhantomData, } } /// Returns the pointer representation of the DebugFS directory. /// /// # Guarantees /// /// Due to the type invariant, the value returned from this function will always be an error /// code, NULL, or a live DebugFS directory. If it is live, it will remain live at least as /// long as this entry lives. pub(crate) fn as_ptr(&self) -> *mut bindings::dentry { self.entry } } impl Drop for Entry<'_> { fn drop(&mut self) { // SAFETY: `debugfs_remove` can take `NULL`, error values, and legal DebugFS dentries. // `as_ptr` guarantees that the pointer is of this form. unsafe { bindings::debugfs_remove(self.as_ptr()) } } }