| // SPDX-License-Identifier: GPL-2.0 |
| |
| // Copyright (C) 2025 Google LLC. |
| |
| use crate::transaction::Transaction; |
| |
| use kernel::bindings::{rust_binder_transaction, task_struct}; |
| use kernel::error::Result; |
| use kernel::ffi::{c_int, c_uint, c_ulong}; |
| use kernel::task::Task; |
| use kernel::tracepoint::declare_trace; |
| |
| declare_trace! { |
| unsafe fn binder_ioctl(cmd: c_uint, arg: c_ulong); |
| unsafe fn binder_ioctl_done(ret: c_int); |
| unsafe fn binder_read_done(ret: c_int); |
| unsafe fn binder_write_done(ret: c_int); |
| unsafe fn binder_wait_for_work(proc_work: bool, transaction_stack: bool, thread_todo: bool); |
| unsafe fn binder_transaction(reply: bool, t: rust_binder_transaction, thread: *mut task_struct); |
| unsafe fn binder_transaction_received(t: rust_binder_transaction); |
| unsafe fn binder_transaction_fd_send(t_debug_id: c_int, fd: c_int, offset: usize); |
| unsafe fn binder_transaction_fd_recv(t_debug_id: c_int, fd: c_int, offset: usize); |
| unsafe fn binder_command(cmd: u32); |
| unsafe fn binder_return(ret: u32); |
| } |
| |
| #[inline] |
| fn raw_transaction(t: &Transaction) -> rust_binder_transaction { |
| t as *const Transaction as rust_binder_transaction |
| } |
| |
| #[inline] |
| fn to_errno(ret: Result) -> i32 { |
| match ret { |
| Ok(()) => 0, |
| Err(err) => err.to_errno(), |
| } |
| } |
| |
| #[inline] |
| pub(crate) fn trace_ioctl(cmd: u32, arg: usize) { |
| // SAFETY: Always safe to call. |
| unsafe { binder_ioctl(cmd, arg as c_ulong) } |
| } |
| |
| #[inline] |
| pub(crate) fn trace_ioctl_done(ret: Result) { |
| // SAFETY: Always safe to call. |
| unsafe { binder_ioctl_done(to_errno(ret)) } |
| } |
| #[inline] |
| pub(crate) fn trace_read_done(ret: Result) { |
| // SAFETY: Always safe to call. |
| unsafe { binder_read_done(to_errno(ret)) } |
| } |
| #[inline] |
| pub(crate) fn trace_write_done(ret: Result) { |
| // SAFETY: Always safe to call. |
| unsafe { binder_write_done(to_errno(ret)) } |
| } |
| |
| #[inline] |
| pub(crate) fn trace_wait_for_work(proc_work: bool, transaction_stack: bool, thread_todo: bool) { |
| // SAFETY: Always safe to call. |
| unsafe { binder_wait_for_work(proc_work, transaction_stack, thread_todo) } |
| } |
| |
| #[inline] |
| pub(crate) fn trace_transaction(reply: bool, t: &Transaction, thread: Option<&Task>) { |
| let thread = match thread { |
| Some(thread) => thread.as_ptr(), |
| None => core::ptr::null_mut(), |
| }; |
| // SAFETY: The raw transaction is valid for the duration of this call. The thread pointer is |
| // valid or null. |
| unsafe { binder_transaction(reply, raw_transaction(t), thread) } |
| } |
| |
| #[inline] |
| pub(crate) fn trace_transaction_received(t: &Transaction) { |
| // SAFETY: The raw transaction is valid for the duration of this call. |
| unsafe { binder_transaction_received(raw_transaction(t)) } |
| } |
| |
| #[inline] |
| pub(crate) fn trace_transaction_fd_send(t_debug_id: usize, fd: u32, offset: usize) { |
| // SAFETY: This function is always safe to call. |
| unsafe { binder_transaction_fd_send(t_debug_id as c_int, fd as c_int, offset) } |
| } |
| #[inline] |
| pub(crate) fn trace_transaction_fd_recv(t_debug_id: usize, fd: u32, offset: usize) { |
| // SAFETY: This function is always safe to call. |
| unsafe { binder_transaction_fd_recv(t_debug_id as c_int, fd as c_int, offset) } |
| } |
| |
| #[inline] |
| pub(crate) fn trace_command(cmd: u32) { |
| // SAFETY: This function is always safe to call. |
| unsafe { binder_command(cmd) } |
| } |
| #[inline] |
| pub(crate) fn trace_return(ret: u32) { |
| // SAFETY: This function is always safe to call. |
| unsafe { binder_return(ret) } |
| } |