You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
aya/test/integration-test/tests/ring_buf.rs

127 lines
3.9 KiB
Rust

use anyhow::{Context as _, Context};
use aya::{include_bytes_aligned, maps::ring_buf::RingBuf, programs::UProbe, Bpf};
use std::os::fd::AsRawFd as _;
use tokio::{
io::unix::AsyncFd,
task::spawn,
time::{sleep, Duration},
};
#[test]
fn ring_buf() {
let bytes = include_bytes_aligned!("../../../target/bpfel-unknown-none/release/ring_buf");
let mut bpf = Bpf::load(bytes).unwrap();
let ring_buf = bpf.take_map("RING_BUF").unwrap();
let mut ring_buf = RingBuf::try_from(ring_buf).unwrap();
let prog: &mut UProbe = bpf
.program_mut("ring_buf_test")
.unwrap()
.try_into()
.unwrap();
prog.load().unwrap();
prog.attach(
Some("ring_buf_trigger_ebpf_program"),
0,
"/proc/self/exe",
None,
)
.unwrap();
// Generate some random data.
let data = gen_data();
// Call the function that the uprobe is attached to with randomly generated data.
for val in &data {
ring_buf_trigger_ebpf_program(*val);
}
// Read the data back out of the ring buffer.
let mut seen = Vec::<u64>::new();
while seen.len() < data.len() {
if let Some(item) = ring_buf.next() {
let item: [u8; 8] = (*item).try_into().unwrap();
let arg = u64::from_ne_bytes(item);
seen.push(arg);
}
}
// Ensure that the data that was read matches what was passed.
assert_eq!(seen, data);
}
#[no_mangle]
#[inline(never)]
pub extern "C" fn ring_buf_trigger_ebpf_program(_arg: u64) {}
/// Generate a variable length vector of u64s. The number of values is always small enough to fit
/// into the RING_BUF defined in the probe.
pub(crate) fn gen_data() -> Vec<u64> {
const DATA_LEN_RANGE: core::ops::RangeInclusive<usize> = 1..=1024;
use rand::Rng as _;
let mut rng = rand::thread_rng();
let n = rng.gen_range(DATA_LEN_RANGE);
std::iter::repeat_with(|| rng.gen()).take(n).collect()
}
#[tokio::test]
async fn ring_buf_async() {
let bytes = include_bytes_aligned!("../../../target/bpfel-unknown-none/release/ring_buf");
let mut bpf = Bpf::load(bytes).unwrap();
let ring_buf = bpf.take_map("RING_BUF").unwrap();
let mut ring_buf = RingBuf::try_from(ring_buf).unwrap();
let prog: &mut UProbe = bpf
.program_mut("ring_buf_test")
.unwrap()
.try_into()
.unwrap();
prog.load().unwrap();
prog.attach(
Some("ring_buf_trigger_ebpf_program"),
0,
"/proc/self/exe",
None,
)
.unwrap();
// Generate some random data.
let data = gen_data();
let write_handle = spawn(call_ring_buf_trigger_ebpf_program_over_time(data.clone()));
// Construct an AsyncFd from the RingBuf in order to receive readiness notifications.
let async_fd = AsyncFd::new(ring_buf.as_raw_fd()).unwrap();
let seen = {
let mut seen = Vec::with_capacity(data.len());
while seen.len() < data.len() {
// Wait for readiness, then clear the bit before reading so that no notifications
// are missed.
async_fd.readable().await.unwrap().clear_ready();
while let Some(data) = ring_buf.next() {
let data: [u8; 8] = (*data)
.try_into()
.context(format!("data: {:?}", (&*data).len()))
.unwrap();
let arg = u64::from_ne_bytes(data);
seen.push(arg);
}
}
seen
};
// Ensure that the data that was read matches what was passed.
assert_eq!(seen, data);
write_handle.await.unwrap();
}
async fn call_ring_buf_trigger_ebpf_program_over_time(data: Vec<u64>) {
let random_duration = || {
use rand::Rng as _;
let mut rng = rand::thread_rng();
let micros = rng.gen_range(0..1_000);
Duration::from_micros(micros)
};
for value in data {
sleep(random_duration()).await;
ring_buf_trigger_ebpf_program(value);
}
}