tracepoint.rs 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. use super::Result;
  2. use crate::bpf::helper::BPF_HELPER_FUN_SET;
  3. use crate::bpf::prog::BpfProg;
  4. use crate::filesystem::page_cache::PageCache;
  5. use crate::libs::casting::DowncastArc;
  6. use crate::libs::spinlock::SpinLock;
  7. use crate::perf::util::PerfProbeConfig;
  8. use crate::perf::{BasicPerfEbpfCallBack, JITMem};
  9. use crate::tracepoint::{TracePoint, TracePointCallBackFunc};
  10. use crate::{
  11. filesystem::vfs::{file::File, FilePrivateData, FileSystem, IndexNode},
  12. libs::spinlock::SpinLockGuard,
  13. perf::{util::PerfProbeArgs, PerfEventOps},
  14. };
  15. use alloc::boxed::Box;
  16. use alloc::sync::Arc;
  17. use alloc::{string::String, vec::Vec};
  18. use core::any::Any;
  19. use core::sync::atomic::AtomicUsize;
  20. use rbpf::EbpfVmRaw;
  21. use system_error::SystemError;
  22. #[derive(Debug)]
  23. pub struct TracepointPerfEvent {
  24. _args: PerfProbeArgs,
  25. tp: &'static TracePoint,
  26. ebpf_list: SpinLock<Vec<usize>>,
  27. }
  28. impl TracepointPerfEvent {
  29. pub fn new(args: PerfProbeArgs, tp: &'static TracePoint) -> TracepointPerfEvent {
  30. TracepointPerfEvent {
  31. _args: args,
  32. tp,
  33. ebpf_list: SpinLock::new(Vec::new()),
  34. }
  35. }
  36. }
  37. impl IndexNode for TracepointPerfEvent {
  38. fn read_at(
  39. &self,
  40. _offset: usize,
  41. _len: usize,
  42. _buf: &mut [u8],
  43. _data: SpinLockGuard<FilePrivateData>,
  44. ) -> Result<usize> {
  45. panic!("read_at not implemented for TracepointPerfEvent");
  46. }
  47. fn write_at(
  48. &self,
  49. _offset: usize,
  50. _len: usize,
  51. _buf: &[u8],
  52. _data: SpinLockGuard<FilePrivateData>,
  53. ) -> Result<usize> {
  54. panic!("write_at not implemented for TracepointPerfEvent");
  55. }
  56. fn fs(&self) -> Arc<dyn FileSystem> {
  57. panic!("fs not implemented for TracepointPerfEvent");
  58. }
  59. fn as_any_ref(&self) -> &dyn Any {
  60. self
  61. }
  62. fn list(&self) -> Result<Vec<String>> {
  63. Err(SystemError::ENOSYS)
  64. }
  65. fn page_cache(&self) -> Option<Arc<PageCache>> {
  66. None
  67. }
  68. }
  69. pub struct TracePointPerfCallBack(BasicPerfEbpfCallBack);
  70. impl TracePointCallBackFunc for TracePointPerfCallBack {
  71. fn call(&self, entry: &[u8]) {
  72. // ebpf needs a mutable slice
  73. let entry =
  74. unsafe { core::slice::from_raw_parts_mut(entry.as_ptr() as *mut u8, entry.len()) };
  75. self.0.call(entry);
  76. }
  77. }
  78. impl PerfEventOps for TracepointPerfEvent {
  79. fn set_bpf_prog(&self, bpf_prog: Arc<File>) -> Result<()> {
  80. static CALLBACK_ID: AtomicUsize = AtomicUsize::new(0);
  81. let file = bpf_prog
  82. .inode()
  83. .downcast_arc::<BpfProg>()
  84. .ok_or(SystemError::EINVAL)?;
  85. let prog_slice = file.insns();
  86. let prog_slice =
  87. unsafe { core::slice::from_raw_parts(prog_slice.as_ptr(), prog_slice.len()) };
  88. let mut vm = EbpfVmRaw::new(Some(prog_slice)).map_err(|e| {
  89. log::error!("create ebpf vm failed: {:?}", e);
  90. SystemError::EINVAL
  91. })?;
  92. for (id, f) in BPF_HELPER_FUN_SET.get() {
  93. vm.register_helper(*id, *f)
  94. .map_err(|_| SystemError::EINVAL)?;
  95. }
  96. // create a callback to execute the ebpf prog
  97. let callback;
  98. #[cfg(target_arch = "x86_64")]
  99. {
  100. log::info!("Using JIT compilation for BPF program on x86_64 architecture");
  101. let jit_mem = Box::new(JITMem::new());
  102. let jit_mem = Box::leak(jit_mem);
  103. let jit_mem_addr = core::ptr::from_ref::<JITMem>(jit_mem) as usize;
  104. vm.set_jit_exec_memory(jit_mem).unwrap();
  105. vm.jit_compile().unwrap();
  106. let basic_callback = BasicPerfEbpfCallBack::new(file, vm, jit_mem_addr);
  107. callback = Box::new(TracePointPerfCallBack(basic_callback));
  108. }
  109. #[cfg(not(target_arch = "x86_64"))]
  110. {
  111. vm.register_allowed_memory(0..u64::MAX);
  112. let basic_callback = BasicPerfEbpfCallBack::new(file, vm);
  113. callback = Box::new(TracePointPerfCallBack(basic_callback));
  114. }
  115. let id = CALLBACK_ID.fetch_add(1, core::sync::atomic::Ordering::Relaxed);
  116. self.tp.register_raw_callback(id, callback);
  117. log::info!(
  118. "Registered BPF program for tracepoint: {}:{} with ID: {}",
  119. self.tp.system(),
  120. self.tp.name(),
  121. id
  122. );
  123. // Store the ID in the ebpf_list for later cleanup
  124. self.ebpf_list.lock().push(id);
  125. Ok(())
  126. }
  127. fn enable(&self) -> Result<()> {
  128. log::info!(
  129. "Enabling tracepoint event: {}:{}",
  130. self.tp.system(),
  131. self.tp.name()
  132. );
  133. self.tp.enable();
  134. Ok(())
  135. }
  136. fn disable(&self) -> Result<()> {
  137. self.tp.disable();
  138. Ok(())
  139. }
  140. fn readable(&self) -> bool {
  141. true
  142. }
  143. }
  144. impl Drop for TracepointPerfEvent {
  145. fn drop(&mut self) {
  146. // Unregister all callbacks associated with this tracepoint event
  147. let mut ebpf_list = self.ebpf_list.lock();
  148. for id in ebpf_list.iter() {
  149. self.tp.unregister_raw_callback(*id);
  150. }
  151. ebpf_list.clear();
  152. }
  153. }
  154. /// Creates a new `TracepointPerfEvent` for the given tracepoint ID.
  155. pub fn perf_event_open_tracepoint(args: PerfProbeArgs) -> Result<TracepointPerfEvent> {
  156. let tp_id = match args.config {
  157. PerfProbeConfig::Raw(tp_id) => tp_id as u32,
  158. _ => {
  159. panic!("Invalid PerfProbeConfig for TracepointPerfEvent");
  160. }
  161. };
  162. let tp_manager = crate::debug::tracing::tracing_events_manager();
  163. let tp_map = tp_manager.tracepoint_map();
  164. let tp = tp_map.get(&tp_id).ok_or(SystemError::ENOENT)?;
  165. Ok(TracepointPerfEvent::new(args, tp))
  166. }