|
@@ -0,0 +1,67 @@
|
|
|
+use core::{marker::PhantomData, mem};
|
|
|
+
|
|
|
+use crate::{
|
|
|
+ bindings::{bpf_map_def, bpf_map_type::BPF_MAP_TYPE_STACK},
|
|
|
+ helpers::{bpf_map_pop_elem, bpf_map_push_elem},
|
|
|
+ maps::PinningType,
|
|
|
+};
|
|
|
+
|
|
|
+#[repr(transparent)]
|
|
|
+pub struct Stack<T> {
|
|
|
+ def: bpf_map_def,
|
|
|
+ _t: PhantomData<T>,
|
|
|
+}
|
|
|
+
|
|
|
+impl<T> Stack<T> {
|
|
|
+ pub const fn with_max_entries(max_entries: u32, flags: u32) -> Stack<T> {
|
|
|
+ Stack {
|
|
|
+ def: bpf_map_def {
|
|
|
+ type_: BPF_MAP_TYPE_STACK,
|
|
|
+ key_size: 0,
|
|
|
+ value_size: mem::size_of::<T>() as u32,
|
|
|
+ max_entries,
|
|
|
+ map_flags: flags,
|
|
|
+ id: 0,
|
|
|
+ pinning: PinningType::None as u32,
|
|
|
+ },
|
|
|
+ _t: PhantomData,
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ pub const fn pinned(max_entries: u32, flags: u32) -> Stack<T> {
|
|
|
+ Stack {
|
|
|
+ def: bpf_map_def {
|
|
|
+ type_: BPF_MAP_TYPE_STACK,
|
|
|
+ key_size: 0,
|
|
|
+ value_size: mem::size_of::<T>() as u32,
|
|
|
+ max_entries,
|
|
|
+ map_flags: flags,
|
|
|
+ id: 0,
|
|
|
+ pinning: PinningType::ByName as u32,
|
|
|
+ },
|
|
|
+ _t: PhantomData,
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn push(&mut self, value: &T, flags: u64) -> Result<(), i64> {
|
|
|
+ let ret = unsafe {
|
|
|
+ bpf_map_push_elem(
|
|
|
+ &mut self.def as *mut _ as *mut _,
|
|
|
+ value as *const _ as *const _,
|
|
|
+ flags,
|
|
|
+ )
|
|
|
+ };
|
|
|
+ (ret == 0).then(|| ()).ok_or(ret)
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn pop(&mut self) -> Option<T> {
|
|
|
+ unsafe {
|
|
|
+ let mut value = mem::MaybeUninit::uninit();
|
|
|
+ let ret = bpf_map_pop_elem(
|
|
|
+ &mut self.def as *mut _ as *mut _,
|
|
|
+ value.as_mut_ptr() as *mut _,
|
|
|
+ );
|
|
|
+ (ret == 0).then(|| value.assume_init())
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|