|  | // SPDX-License-Identifier: GPL-2.0 | 
|  |  | 
|  | // Copyright (C) 2024 Google LLC. | 
|  |  | 
|  | //! Logic for tracepoints. | 
|  |  | 
|  | /// Declare the Rust entry point for a tracepoint. | 
|  | /// | 
|  | /// This macro generates an unsafe function that calls into C, and its safety requirements will be | 
|  | /// whatever the relevant C code requires. To document these safety requirements, you may add | 
|  | /// doc-comments when invoking the macro. | 
|  | #[macro_export] | 
|  | macro_rules! declare_trace { | 
|  | ($($(#[$attr:meta])* $pub:vis unsafe fn $name:ident($($argname:ident : $argtyp:ty),* $(,)?);)*) => {$( | 
|  | $( #[$attr] )* | 
|  | #[inline(always)] | 
|  | $pub unsafe fn $name($($argname : $argtyp),*) { | 
|  | #[cfg(CONFIG_TRACEPOINTS)] | 
|  | { | 
|  | // SAFETY: It's always okay to query the static key for a tracepoint. | 
|  | let should_trace = unsafe { | 
|  | $crate::macros::paste! { | 
|  | $crate::jump_label::static_branch_unlikely!( | 
|  | $crate::bindings::[< __tracepoint_ $name >], | 
|  | $crate::bindings::tracepoint, | 
|  | key | 
|  | ) | 
|  | } | 
|  | }; | 
|  |  | 
|  | if should_trace { | 
|  | $crate::macros::paste! { | 
|  | // SAFETY: The caller guarantees that it is okay to call this tracepoint. | 
|  | unsafe { $crate::bindings::[< rust_do_trace_ $name >]($($argname),*) }; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | #[cfg(not(CONFIG_TRACEPOINTS))] | 
|  | { | 
|  | // If tracepoints are disabled, insert a trivial use of each argument | 
|  | // to avoid unused argument warnings. | 
|  | $( let _unused = $argname; )* | 
|  | } | 
|  | } | 
|  | )*} | 
|  | } | 
|  |  | 
|  | pub use declare_trace; |