sk_skb.rs 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. use std::borrow::Cow;
  2. use proc_macro2::TokenStream;
  3. use proc_macro_error::abort;
  4. use quote::quote;
  5. use syn::{ItemFn, Result};
  6. #[allow(clippy::enum_variant_names)]
  7. #[derive(Debug, Copy, Clone)]
  8. pub(crate) enum SkSkbKind {
  9. StreamVerdict,
  10. StreamParser,
  11. }
  12. impl std::fmt::Display for SkSkbKind {
  13. fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
  14. use SkSkbKind::*;
  15. match self {
  16. StreamVerdict => write!(f, "stream_verdict"),
  17. StreamParser => write!(f, "stream_parser"),
  18. }
  19. }
  20. }
  21. pub(crate) struct SkSkb {
  22. kind: SkSkbKind,
  23. item: ItemFn,
  24. }
  25. impl SkSkb {
  26. pub(crate) fn parse(kind: SkSkbKind, attrs: TokenStream, item: TokenStream) -> Result<SkSkb> {
  27. if !attrs.is_empty() {
  28. abort!(attrs, "unexpected attribute");
  29. }
  30. let item = syn::parse2(item)?;
  31. Ok(SkSkb { item, kind })
  32. }
  33. pub(crate) fn expand(&self) -> Result<TokenStream> {
  34. let kind = &self.kind;
  35. let section_name: Cow<'_, _> = format!("sk_skb/{kind}").into();
  36. let fn_name = self.item.sig.ident.clone();
  37. let fn_vis = &self.item.vis;
  38. let item = &self.item;
  39. Ok(quote! {
  40. #[no_mangle]
  41. #[link_section = #section_name]
  42. #fn_vis fn #fn_name(ctx: *mut ::aya_ebpf::bindings::__sk_buff) -> u32 {
  43. return #fn_name(::aya_ebpf::programs::SkBuffContext::new(ctx));
  44. #item
  45. }
  46. })
  47. }
  48. }
  49. #[cfg(test)]
  50. mod tests {
  51. use syn::parse_quote;
  52. use super::*;
  53. #[test]
  54. fn test_stream_parser() {
  55. let prog = SkSkb::parse(
  56. SkSkbKind::StreamParser,
  57. parse_quote! {},
  58. parse_quote! {
  59. fn prog(ctx: &mut ::aya_ebpf::programs::SkBuffContext) -> u32 {
  60. 0
  61. }
  62. },
  63. )
  64. .unwrap();
  65. let expanded = prog.expand().unwrap();
  66. let expected = quote! {
  67. #[no_mangle]
  68. #[link_section = "sk_skb/stream_parser"]
  69. fn prog(ctx: *mut ::aya_ebpf::bindings::__sk_buff) -> u32 {
  70. return prog(::aya_ebpf::programs::SkBuffContext::new(ctx));
  71. fn prog(ctx: &mut ::aya_ebpf::programs::SkBuffContext) -> u32 {
  72. 0
  73. }
  74. }
  75. };
  76. assert_eq!(expected.to_string(), expanded.to_string());
  77. }
  78. #[test]
  79. fn test_stream_verdict() {
  80. let prog = SkSkb::parse(
  81. SkSkbKind::StreamVerdict,
  82. parse_quote! {},
  83. parse_quote! {
  84. fn prog(ctx: &mut ::aya_ebpf::programs::SkBuffContext) -> u32 {
  85. 0
  86. }
  87. },
  88. )
  89. .unwrap();
  90. let expanded = prog.expand().unwrap();
  91. let expected = quote! {
  92. #[no_mangle]
  93. #[link_section = "sk_skb/stream_verdict"]
  94. fn prog(ctx: *mut ::aya_ebpf::bindings::__sk_buff) -> u32 {
  95. return prog(::aya_ebpf::programs::SkBuffContext::new(ctx));
  96. fn prog(ctx: &mut ::aya_ebpf::programs::SkBuffContext) -> u32 {
  97. 0
  98. }
  99. }
  100. };
  101. assert_eq!(expected.to_string(), expanded.to_string());
  102. }
  103. }