4
0

cgroup_skb.rs 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214
  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::{Ident, ItemFn};
  6. pub(crate) struct CgroupSkb {
  7. item: ItemFn,
  8. attach_type: Option<Ident>,
  9. }
  10. impl CgroupSkb {
  11. pub(crate) fn parse(attrs: TokenStream, item: TokenStream) -> Result<Self, Diagnostic> {
  12. let item: ItemFn = syn::parse2(item)?;
  13. let attach_type = if attrs.is_empty() {
  14. None
  15. } else {
  16. let ident: Ident = syn::parse2(attrs)?;
  17. if ident != "ingress" && ident != "egress" {
  18. return Err(ident.span().error("invalid attach type"));
  19. }
  20. Some(ident)
  21. };
  22. Ok(Self { item, attach_type })
  23. }
  24. pub(crate) fn expand(&self) -> TokenStream {
  25. let Self { item, attach_type } = self;
  26. let ItemFn {
  27. attrs: _,
  28. vis,
  29. sig,
  30. block: _,
  31. } = item;
  32. let section_name: Cow<'_, _> = match attach_type {
  33. Some(attach_type) => format!("cgroup_skb/{attach_type}").into(),
  34. None => "cgroup/skb".into(),
  35. };
  36. let fn_name = &sig.ident;
  37. quote! {
  38. #[no_mangle]
  39. #[link_section = #section_name]
  40. #vis fn #fn_name(ctx: *mut ::aya_ebpf::bindings::__sk_buff) -> i32 {
  41. return #fn_name(::aya_ebpf::programs::SkBuffContext::new(ctx));
  42. #item
  43. }
  44. }
  45. }
  46. }
  47. #[cfg(test)]
  48. mod tests {
  49. use syn::parse_quote;
  50. use super::*;
  51. #[test]
  52. fn cgroup_skb() {
  53. let prog = CgroupSkb::parse(
  54. parse_quote!(),
  55. parse_quote!(
  56. fn foo(ctx: SkBuffContext) -> i32 {
  57. 0
  58. }
  59. ),
  60. )
  61. .unwrap();
  62. let expanded = prog.expand();
  63. let expected = quote! {
  64. #[no_mangle]
  65. #[link_section = "cgroup/skb"]
  66. fn foo(ctx: *mut ::aya_ebpf::bindings::__sk_buff) -> i32 {
  67. return foo(::aya_ebpf::programs::SkBuffContext::new(ctx));
  68. fn foo(ctx: SkBuffContext) -> i32 {
  69. 0
  70. }
  71. }
  72. };
  73. assert_eq!(expected.to_string(), expanded.to_string());
  74. }
  75. #[test]
  76. fn cgroup_skb_egress() {
  77. let prog = CgroupSkb::parse(
  78. parse_quote!(egress),
  79. parse_quote!(
  80. fn foo(ctx: SkBuffContext) -> i32 {
  81. 0
  82. }
  83. ),
  84. )
  85. .unwrap();
  86. let expanded = prog.expand();
  87. let expected = quote! {
  88. #[no_mangle]
  89. #[link_section = "cgroup_skb/egress"]
  90. fn foo(ctx: *mut ::aya_ebpf::bindings::__sk_buff) -> i32 {
  91. return foo(::aya_ebpf::programs::SkBuffContext::new(ctx));
  92. fn foo(ctx: SkBuffContext) -> i32 {
  93. 0
  94. }
  95. }
  96. };
  97. assert_eq!(expected.to_string(), expanded.to_string());
  98. }
  99. #[test]
  100. fn cgroup_skb_ingress() {
  101. let prog = CgroupSkb::parse(
  102. parse_quote!(ingress),
  103. parse_quote!(
  104. fn foo(ctx: SkBuffContext) -> i32 {
  105. 0
  106. }
  107. ),
  108. )
  109. .unwrap();
  110. let expanded = prog.expand();
  111. let expected = quote! {
  112. #[no_mangle]
  113. #[link_section = "cgroup_skb/ingress"]
  114. fn foo(ctx: *mut ::aya_ebpf::bindings::__sk_buff) -> i32 {
  115. return foo(::aya_ebpf::programs::SkBuffContext::new(ctx));
  116. fn foo(ctx: SkBuffContext) -> i32 {
  117. 0
  118. }
  119. }
  120. };
  121. assert_eq!(expected.to_string(), expanded.to_string());
  122. }
  123. #[test]
  124. fn priv_function() {
  125. let prog = CgroupSkb::parse(
  126. parse_quote!(egress),
  127. parse_quote!(
  128. fn foo(ctx: SkBuffContext) -> i32 {
  129. 0
  130. }
  131. ),
  132. )
  133. .unwrap();
  134. let expanded = prog.expand();
  135. let expected = quote! {
  136. #[no_mangle]
  137. #[link_section = "cgroup_skb/egress"]
  138. fn foo(ctx: *mut ::aya_ebpf::bindings::__sk_buff) -> i32 {
  139. return foo(::aya_ebpf::programs::SkBuffContext::new(ctx));
  140. fn foo(ctx: SkBuffContext) -> i32 {
  141. 0
  142. }
  143. }
  144. };
  145. assert_eq!(expected.to_string(), expanded.to_string());
  146. }
  147. #[test]
  148. fn pub_function() {
  149. let prog = CgroupSkb::parse(
  150. parse_quote!(egress),
  151. parse_quote!(
  152. pub fn foo(ctx: SkBuffContext) -> i32 {
  153. 0
  154. }
  155. ),
  156. )
  157. .unwrap();
  158. let expanded = prog.expand();
  159. let expected = quote! {
  160. #[no_mangle]
  161. #[link_section = "cgroup_skb/egress"]
  162. pub fn foo(ctx: *mut ::aya_ebpf::bindings::__sk_buff) -> i32 {
  163. return foo(::aya_ebpf::programs::SkBuffContext::new(ctx));
  164. pub fn foo(ctx: SkBuffContext) -> i32 {
  165. 0
  166. }
  167. }
  168. };
  169. assert_eq!(expected.to_string(), expanded.to_string());
  170. }
  171. #[test]
  172. fn pub_crate_function() {
  173. let prog = CgroupSkb::parse(
  174. parse_quote!(egress),
  175. parse_quote!(
  176. pub(crate) fn foo(ctx: SkBuffContext) -> i32 {
  177. 0
  178. }
  179. ),
  180. )
  181. .unwrap();
  182. let expanded = prog.expand();
  183. let expected = quote! {
  184. #[no_mangle]
  185. #[link_section = "cgroup_skb/egress"]
  186. pub(crate) fn foo(ctx: *mut ::aya_ebpf::bindings::__sk_buff) -> i32 {
  187. return foo(::aya_ebpf::programs::SkBuffContext::new(ctx));
  188. pub(crate) fn foo(ctx: SkBuffContext) -> i32 {
  189. 0
  190. }
  191. }
  192. };
  193. assert_eq!(expected.to_string(), expanded.to_string());
  194. }
  195. }