sk_skb.rs 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  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. #[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(
  27. kind: SkSkbKind,
  28. attrs: TokenStream,
  29. item: TokenStream,
  30. ) -> Result<Self, Diagnostic> {
  31. if !attrs.is_empty() {
  32. return Err(attrs.span().error("unexpected attribute"));
  33. }
  34. let item = syn::parse2(item)?;
  35. Ok(Self { item, kind })
  36. }
  37. pub(crate) fn expand(&self) -> TokenStream {
  38. let Self { kind, item } = self;
  39. let ItemFn {
  40. attrs: _,
  41. vis,
  42. sig,
  43. block: _,
  44. } = item;
  45. let section_name: Cow<'_, _> = format!("sk_skb/{kind}").into();
  46. let fn_name = &sig.ident;
  47. quote! {
  48. #[no_mangle]
  49. #[link_section = #section_name]
  50. #vis fn #fn_name(ctx: *mut ::aya_ebpf::bindings::__sk_buff) -> u32 {
  51. return #fn_name(::aya_ebpf::programs::SkBuffContext::new(ctx));
  52. #item
  53. }
  54. }
  55. }
  56. }
  57. #[cfg(test)]
  58. mod tests {
  59. use syn::parse_quote;
  60. use super::*;
  61. #[test]
  62. fn test_stream_parser() {
  63. let prog = SkSkb::parse(
  64. SkSkbKind::StreamParser,
  65. parse_quote! {},
  66. parse_quote! {
  67. fn prog(ctx: &mut ::aya_ebpf::programs::SkBuffContext) -> u32 {
  68. 0
  69. }
  70. },
  71. )
  72. .unwrap();
  73. let expanded = prog.expand();
  74. let expected = quote! {
  75. #[no_mangle]
  76. #[link_section = "sk_skb/stream_parser"]
  77. fn prog(ctx: *mut ::aya_ebpf::bindings::__sk_buff) -> u32 {
  78. return prog(::aya_ebpf::programs::SkBuffContext::new(ctx));
  79. fn prog(ctx: &mut ::aya_ebpf::programs::SkBuffContext) -> u32 {
  80. 0
  81. }
  82. }
  83. };
  84. assert_eq!(expected.to_string(), expanded.to_string());
  85. }
  86. #[test]
  87. fn test_stream_verdict() {
  88. let prog = SkSkb::parse(
  89. SkSkbKind::StreamVerdict,
  90. parse_quote! {},
  91. parse_quote! {
  92. fn prog(ctx: &mut ::aya_ebpf::programs::SkBuffContext) -> u32 {
  93. 0
  94. }
  95. },
  96. )
  97. .unwrap();
  98. let expanded = prog.expand();
  99. let expected = quote! {
  100. #[no_mangle]
  101. #[link_section = "sk_skb/stream_verdict"]
  102. fn prog(ctx: *mut ::aya_ebpf::bindings::__sk_buff) -> u32 {
  103. return prog(::aya_ebpf::programs::SkBuffContext::new(ctx));
  104. fn prog(ctx: &mut ::aya_ebpf::programs::SkBuffContext) -> u32 {
  105. 0
  106. }
  107. }
  108. };
  109. assert_eq!(expected.to_string(), expanded.to_string());
  110. }
  111. }