tracepoint.rs 2.8 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697
  1. use std::borrow::Cow;
  2. use proc_macro2::TokenStream;
  3. use proc_macro2_diagnostics::{Diagnostic, SpanDiagnosticExt as _};
  4. use quote::quote;
  5. use syn::{spanned::Spanned as _, ItemFn};
  6. use crate::args::{err_on_unknown_args, pop_string_arg};
  7. pub(crate) struct TracePoint {
  8. item: ItemFn,
  9. name_and_category: Option<(String, String)>,
  10. }
  11. impl TracePoint {
  12. pub(crate) fn parse(attrs: TokenStream, item: TokenStream) -> Result<Self, Diagnostic> {
  13. let item = syn::parse2(item)?;
  14. let span = attrs.span();
  15. let mut args = syn::parse2(attrs)?;
  16. let name = pop_string_arg(&mut args, "name");
  17. let category = pop_string_arg(&mut args, "category");
  18. err_on_unknown_args(&args)?;
  19. match (name, category) {
  20. (None, None) => Ok(Self {
  21. item,
  22. name_and_category: None,
  23. }),
  24. (Some(name), Some(category)) => Ok(Self {
  25. item,
  26. name_and_category: Some((name, category)),
  27. }),
  28. _ => Err(span.error("expected `name` and `category` arguments")),
  29. }
  30. }
  31. pub(crate) fn expand(&self) -> TokenStream {
  32. let Self {
  33. item,
  34. name_and_category,
  35. } = self;
  36. let section_name: Cow<'_, _> = match name_and_category {
  37. Some((name, category)) => format!("tracepoint/{category}/{name}").into(),
  38. None => "tracepoint".into(),
  39. };
  40. let ItemFn {
  41. attrs: _,
  42. vis,
  43. sig,
  44. block: _,
  45. } = item;
  46. let fn_name = &sig.ident;
  47. quote! {
  48. #[no_mangle]
  49. #[link_section = #section_name]
  50. #vis fn #fn_name(ctx: *mut ::core::ffi::c_void) -> u32 {
  51. let _ = #fn_name(::aya_ebpf::programs::TracePointContext::new(ctx));
  52. return 0;
  53. #item
  54. }
  55. }
  56. }
  57. }
  58. #[cfg(test)]
  59. mod tests {
  60. use syn::parse_quote;
  61. use super::*;
  62. #[test]
  63. fn test_tracepoint() {
  64. let prog = TracePoint::parse(
  65. parse_quote! { name = "sys_enter_bind", category = "syscalls" },
  66. parse_quote! {
  67. fn prog(ctx: &mut ::aya_ebpf::programs::TracePointContext) -> i32 {
  68. 0
  69. }
  70. },
  71. )
  72. .unwrap();
  73. let expanded = prog.expand();
  74. let expected = quote! {
  75. #[no_mangle]
  76. #[link_section = "tracepoint/syscalls/sys_enter_bind"]
  77. fn prog(ctx: *mut ::core::ffi::c_void) -> u32 {
  78. let _ = prog(::aya_ebpf::programs::TracePointContext::new(ctx));
  79. return 0;
  80. fn prog(ctx: &mut ::aya_ebpf::programs::TracePointContext) -> i32 {
  81. 0
  82. }
  83. }
  84. };
  85. assert_eq!(expected.to_string(), expanded.to_string());
  86. }
  87. }