dev_map.rs 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. use core::{cell::UnsafeCell, mem, num::NonZeroU32};
  2. use aya_ebpf_bindings::bindings::bpf_devmap_val;
  3. use super::try_redirect_map;
  4. use crate::{
  5. bindings::{bpf_map_def, bpf_map_type::BPF_MAP_TYPE_DEVMAP},
  6. lookup,
  7. maps::PinningType,
  8. };
  9. /// An array of network devices.
  10. ///
  11. /// XDP programs can use this map to redirect packets to other network deviecs.
  12. ///
  13. /// # Minimum kernel version
  14. ///
  15. /// The minimum kernel version required to use this feature is 4.14.
  16. ///
  17. /// # Examples
  18. ///
  19. /// ```rust,no_run
  20. /// use aya_ebpf::{bindings::xdp_action, macros::{map, xdp}, maps::DevMap, programs::XdpContext};
  21. ///
  22. /// #[map]
  23. /// static MAP: DevMap = DevMap::with_max_entries(1, 0);
  24. ///
  25. /// #[xdp]
  26. /// fn xdp(_ctx: XdpContext) -> u32 {
  27. /// MAP.redirect(0, xdp_action::XDP_PASS as u64).unwrap_or(xdp_action::XDP_DROP)
  28. /// }
  29. /// ```
  30. #[repr(transparent)]
  31. pub struct DevMap {
  32. def: UnsafeCell<bpf_map_def>,
  33. }
  34. unsafe impl Sync for DevMap {}
  35. impl DevMap {
  36. /// Creates a [`DevMap`] with a set maximum number of elements.
  37. ///
  38. /// # Examples
  39. ///
  40. /// ```rust,no_run
  41. /// use aya_ebpf::{macros::map, maps::DevMap};
  42. ///
  43. /// #[map]
  44. /// static MAP: DevMap = DevMap::with_max_entries(8, 0);
  45. /// ```
  46. pub const fn with_max_entries(max_entries: u32, flags: u32) -> DevMap {
  47. DevMap {
  48. def: UnsafeCell::new(bpf_map_def {
  49. type_: BPF_MAP_TYPE_DEVMAP,
  50. key_size: mem::size_of::<u32>() as u32,
  51. value_size: mem::size_of::<bpf_devmap_val>() as u32,
  52. max_entries,
  53. map_flags: flags,
  54. id: 0,
  55. pinning: PinningType::None as u32,
  56. }),
  57. }
  58. }
  59. /// Creates a [`DevMap`] with a set maximum number of elements that can be pinned to the BPF
  60. /// File System (bpffs).
  61. ///
  62. /// # Examples
  63. ///
  64. /// ```rust,no_run
  65. /// use aya_ebpf::{macros::map, maps::DevMap};
  66. ///
  67. /// #[map]
  68. /// static MAP: DevMap = DevMap::pinned(8, 0);
  69. /// ```
  70. pub const fn pinned(max_entries: u32, flags: u32) -> DevMap {
  71. DevMap {
  72. def: UnsafeCell::new(bpf_map_def {
  73. type_: BPF_MAP_TYPE_DEVMAP,
  74. key_size: mem::size_of::<u32>() as u32,
  75. value_size: mem::size_of::<bpf_devmap_val>() as u32,
  76. max_entries,
  77. map_flags: flags,
  78. id: 0,
  79. pinning: PinningType::ByName as u32,
  80. }),
  81. }
  82. }
  83. /// Retrieves the interface index at `index` in the array.
  84. ///
  85. /// To actually redirect a packet, see [`DevMap::redirect`].
  86. ///
  87. /// # Examples
  88. ///
  89. /// ```rust,no_run
  90. /// use aya_ebpf::{macros::map, maps::DevMap};
  91. ///
  92. /// #[map]
  93. /// static MAP: DevMap = DevMap::with_max_entries(1, 0);
  94. ///
  95. /// let target_if_index = MAP.get(0).unwrap().if_index;
  96. ///
  97. /// // redirect to if_index
  98. /// ```
  99. #[inline(always)]
  100. pub fn get(&self, index: u32) -> Option<DevMapValue> {
  101. let value = lookup(self.def.get(), &index)?;
  102. let value: &bpf_devmap_val = unsafe { value.as_ref() };
  103. Some(DevMapValue {
  104. if_index: value.ifindex,
  105. // SAFETY: map writes use fd, map reads use id.
  106. // https://elixir.bootlin.com/linux/v6.2/source/include/uapi/linux/bpf.h#L6136
  107. prog_id: NonZeroU32::new(unsafe { value.bpf_prog.id }),
  108. })
  109. }
  110. /// Redirects the current packet on the interface at `index`.
  111. ///
  112. /// The lower two bits of `flags` are used for the return code if the map lookup fails, which
  113. /// can be used as the XDP program's return code if a CPU cannot be found.
  114. ///
  115. /// # Examples
  116. ///
  117. /// ```rust,no_run
  118. /// use aya_ebpf::{bindings::xdp_action, macros::{map, xdp}, maps::DevMap, programs::XdpContext};
  119. ///
  120. /// #[map]
  121. /// static MAP: DevMap = DevMap::with_max_entries(8, 0);
  122. ///
  123. /// #[xdp]
  124. /// fn xdp(_ctx: XdpContext) -> u32 {
  125. /// MAP.redirect(7, 0).unwrap_or(xdp_action::XDP_DROP)
  126. /// }
  127. /// ```
  128. #[inline(always)]
  129. pub fn redirect(&self, index: u32, flags: u64) -> Result<u32, u32> {
  130. try_redirect_map(&self.def, index, flags)
  131. }
  132. }
  133. #[derive(Clone, Copy)]
  134. /// The value of a device map.
  135. pub struct DevMapValue {
  136. /// Target interface index to redirect to.
  137. pub if_index: u32,
  138. /// Chained XDP program ID.
  139. pub prog_id: Option<NonZeroU32>,
  140. }