mod common;
mod test_netfilter;
mod test_print;
mod test_streams;
mod test_tc;
mod test_xdp;
use std::collections::HashMap;
use std::collections::HashSet;
use std::env::current_exe;
use std::ffi::c_int;
use std::ffi::c_void;
use std::ffi::CString;
use std::ffi::OsStr;
use std::fs;
use std::hint;
use std::io;
use std::io::Read;
use std::mem::size_of;
use std::mem::size_of_val;
use std::os::unix::io::AsFd;
use std::os::unix::io::AsRawFd as _;
use std::os::unix::io::FromRawFd as _;
use std::os::unix::io::OwnedFd;
use std::os::unix::io::RawFd;
use std::path::Path;
use std::path::PathBuf;
use std::ptr;
use std::ptr::addr_of;
use std::slice;
use std::sync::atomic::AtomicI32;
use std::sync::atomic::Ordering;
use std::sync::mpsc::channel;
use std::time::Duration;
use libbpf_rs::num_possible_cpus;
use libbpf_rs::query::LinkTypeInfo;
use libbpf_rs::query::PerfEventType;
use libbpf_rs::query::ProgInfoIter;
use libbpf_rs::AsRawLibbpf;
use libbpf_rs::Iter;
use libbpf_rs::KprobeMultiOpts;
use libbpf_rs::KprobeOpts;
use libbpf_rs::Linker;
use libbpf_rs::MapCore;
use libbpf_rs::MapFlags;
use libbpf_rs::MapHandle;
use libbpf_rs::MapInfo;
use libbpf_rs::MapType;
use libbpf_rs::Object;
use libbpf_rs::ObjectBuilder;
use libbpf_rs::PerfEventOpts;
use libbpf_rs::Program;
use libbpf_rs::ProgramInput;
use libbpf_rs::ProgramType;
use libbpf_rs::RawTracepointOpts;
use libbpf_rs::TracepointCategory;
use libbpf_rs::TracepointOpts;
use libbpf_rs::UprobeMultiOpts;
use libbpf_rs::UprobeOpts;
use libbpf_rs::UsdtOpts;
use libbpf_rs::UserRingBuffer;
use plain::Plain;
use probe::probe;
use scopeguard::defer;
use tempfile::NamedTempFile;
use test_tag::tag;
use crate::common::get_map;
use crate::common::get_map_mut;
use crate::common::get_prog_mut;
use crate::common::get_symbol_offset;
use crate::common::get_test_object;
use crate::common::get_test_object_path;
use crate::common::open_test_object;
use crate::common::with_ringbuffer;
#[tag(root)]
#[test]
fn test_object_build_and_load() {
get_test_object("runqslower.bpf.o");
}
#[test]
fn test_object_build_from_memory() {
let obj_path = get_test_object_path("runqslower.bpf.o");
let contents = fs::read(obj_path).expect("failed to read object file");
let mut builder = ObjectBuilder::default();
let obj = builder
.name("memory name")
.unwrap()
.open_memory(&contents)
.expect("failed to build object");
let name = obj.name().expect("failed to get object name");
assert!(name == "memory name");
let obj = unsafe { Object::from_ptr(obj.take_ptr()) };
let name = obj.name().expect("failed to get object name");
assert!(name == "memory name");
}
#[test]
fn test_object_build_from_memory_empty_name() {
let obj_path = get_test_object_path("runqslower.bpf.o");
let contents = fs::read(obj_path).expect("failed to read object file");
let mut builder = ObjectBuilder::default();
let obj = builder
.name("")
.unwrap()
.open_memory(&contents)
.expect("failed to build object");
let name = obj.name().expect("failed to get object name");
assert!(name.is_empty());
let obj = unsafe { Object::from_ptr(obj.take_ptr()) };
let name = obj.name().expect("failed to get object name");
assert!(name.is_empty());
}
#[tag(root)]
#[test]
fn test_object_load_invalid() {
let empty_file = NamedTempFile::new().unwrap();
let _err = ObjectBuilder::default()
.debug(true)
.open_file(empty_file.path())
.unwrap_err();
}
#[test]
fn test_object_name() {
let obj_path = get_test_object_path("runqslower.bpf.o");
let mut builder = ObjectBuilder::default();
builder.name("test name").unwrap();
let obj = builder.open_file(obj_path).expect("failed to build object");
let obj_name = obj.name().expect("failed to get object name");
assert!(obj_name == "test name");
}
#[tag(root)]
#[test]
fn test_valid_btf_custom_path() {
let obj_path = get_test_object_path("runqslower.bpf.o");
let mut builder = ObjectBuilder::default();
builder.btf_custom_path("/sys/kernel/btf/vmlinux").unwrap();
let obj = builder.open_file(obj_path).expect("failed to build object");
obj.load().expect("failed to load object");
}
#[tag(root)]
#[test]
fn test_invalid_btf_custom_path() {
let obj_path = get_test_object_path("runqslower.bpf.o");
let mut builder = ObjectBuilder::default();
builder.btf_custom_path("/").unwrap();
let obj = builder.open_file(obj_path).expect("failed to build object");
assert!(obj.load().is_err());
}
#[tag(root)]
#[test]
fn test_object_maps() {
let mut obj = get_test_object("runqslower.bpf.o");
let _map = get_map_mut(&mut obj, "start");
let _map = get_map_mut(&mut obj, "events");
assert!(!obj.maps().any(|map| map.name() == OsStr::new("asdf")));
}
#[tag(root)]
#[test]
fn test_object_maps_iter() {
let obj = get_test_object("runqslower.bpf.o");
for map in obj.maps() {
eprintln!("{:?}", map.name());
}
assert!(obj.maps().count() == 4);
}
#[tag(root)]
#[test]
fn test_object_map_key_value_size() {
let mut obj = get_test_object("runqslower.bpf.o");
let start = get_map_mut(&mut obj, "start");
assert!(start.lookup(&[1, 2, 3, 4, 5], MapFlags::empty()).is_err());
assert!(start.delete(&[1]).is_err());
assert!(start.lookup_and_delete(&[1, 2, 3, 4, 5]).is_err());
assert!(start
.update(&[1, 2, 3, 4, 5], &[1], MapFlags::empty())
.is_err());
}
#[tag(root)]
#[test]
fn test_object_map_update_batch() {
let mut obj = get_test_object("runqslower.bpf.o");
let start = get_map_mut(&mut obj, "start");
let key1 = 1u32.to_ne_bytes();
let key2 = 2u32.to_ne_bytes();
let key3 = 3u32.to_ne_bytes();
let key4 = 4u32.to_ne_bytes();
let value1 = 369u64.to_ne_bytes();
let value2 = 258u64.to_ne_bytes();
let value3 = 147u64.to_ne_bytes();
let value4 = 159u64.to_ne_bytes();
let batch_key1 = key1.into_iter().chain(key2).collect::<Vec<_>>();
let batch_value1 = value1.into_iter().chain(value2).collect::<Vec<_>>();
let batch_key2 = key2.into_iter().chain(key3).chain(key4).collect::<Vec<_>>();
let batch_value2 = value2
.into_iter()
.chain(value3)
.chain(value4)
.collect::<Vec<_>>();
assert!(start
.update_batch(
&[1, 2, 3],
&batch_value1,
2,
MapFlags::ANY,
MapFlags::NO_EXIST
)
.is_err());
assert!(start
.update_batch(
&batch_key1,
&[1, 2, 3],
2,
MapFlags::ANY,
MapFlags::NO_EXIST
)
.is_err());
assert!(start
.update_batch(
&batch_key1,
&batch_value1,
1,
MapFlags::ANY,
MapFlags::NO_EXIST
)
.is_err());
assert!(start
.update_batch(&key1, &value1, 1, MapFlags::ANY, MapFlags::NO_EXIST)
.is_ok());
assert!(start
.update_batch(
&batch_key2,
&batch_value2,
3,
MapFlags::ANY,
MapFlags::NO_EXIST
)
.is_ok());
assert!(start
.update_batch(
&batch_key2,
&batch_value2,
3,
MapFlags::NO_EXIST,
MapFlags::NO_EXIST
)
.is_err());
}
#[tag(root)]
#[test]
fn test_object_map_lookup_batch() {
let mut obj = get_test_object("runqslower.bpf.o");
let start = get_map_mut(&mut obj, "start");
let data = HashMap::from([
(1u32, 9999u64),
(2u32, 42u64),
(3u32, 18u64),
(4u32, 1337u64),
]);
for (key, val) in data.iter() {
assert!(start
.update(&key.to_ne_bytes(), &val.to_ne_bytes(), MapFlags::ANY)
.is_ok());
}
let elems = start
.lookup_batch(2, MapFlags::ANY, MapFlags::ANY)
.expect("failed to lookup batch")
.collect::<Vec<_>>();
assert_eq!(elems.len(), 4);
for (key, val) in elems.into_iter() {
let key = u32::from_ne_bytes(key.try_into().unwrap());
let val = u64::from_ne_bytes(val.try_into().unwrap());
assert_eq!(val, data[&key]);
}
let elems = start
.lookup_batch(5, MapFlags::ANY, MapFlags::ANY)
.expect("failed to lookup batch")
.collect::<Vec<_>>();
assert_eq!(elems.len(), 4);
for (key, val) in elems.into_iter() {
let key = u32::from_ne_bytes(key.try_into().unwrap());
let val = u64::from_ne_bytes(val.try_into().unwrap());
assert_eq!(val, data[&key]);
}
let elems = start
.lookup_and_delete_batch(3, MapFlags::ANY, MapFlags::ANY)
.expect("failed to lookup batch")
.collect::<Vec<_>>();
assert_eq!(elems.len(), 4);
for (key, val) in elems.into_iter() {
let key = u32::from_ne_bytes(key.try_into().unwrap());
let val = u64::from_ne_bytes(val.try_into().unwrap());
assert_eq!(val, data[&key]);
}
assert!(start.keys().collect::<Vec<_>>().is_empty())
}
#[tag(root)]
#[test]
fn test_object_map_delete_batch() {
let mut obj = get_test_object("runqslower.bpf.o");
let start = get_map_mut(&mut obj, "start");
let key1 = 1u32.to_ne_bytes();
assert!(start
.update(&key1, &9999u64.to_ne_bytes(), MapFlags::ANY)
.is_ok());
let key2 = 2u32.to_ne_bytes();
assert!(start
.update(&key2, &42u64.to_ne_bytes(), MapFlags::ANY)
.is_ok());
let key3 = 3u32.to_ne_bytes();
assert!(start
.update(&key3, &18u64.to_ne_bytes(), MapFlags::ANY)
.is_ok());
let key4 = 4u32.to_ne_bytes();
assert!(start
.update(&key4, &1337u64.to_ne_bytes(), MapFlags::ANY)
.is_ok());
assert!(start
.delete_batch(&[0, 0, 1], 1, MapFlags::empty(), MapFlags::empty())
.is_err());
assert!(start
.delete_batch(&key4, 2, MapFlags::empty(), MapFlags::empty())
.is_err());
assert!(start
.delete_batch(&key4, 1, MapFlags::empty(), MapFlags::empty())
.is_ok());
let keys = key1.into_iter().chain(key2).chain(key3).collect::<Vec<_>>();
assert!(start
.delete_batch(&keys, 3, MapFlags::empty(), MapFlags::empty())
.is_ok());
assert!(start.keys().collect::<Vec<_>>().is_empty())
}
#[tag(root)]
#[test]
pub fn test_map_info() {
let opts = libbpf_sys::bpf_map_create_opts {
sz: size_of::<libbpf_sys::bpf_map_create_opts>() as libbpf_sys::size_t,
map_flags: libbpf_sys::BPF_ANY,
btf_fd: 0,
btf_key_type_id: 0,
btf_value_type_id: 0,
btf_vmlinux_value_type_id: 0,
inner_map_fd: 0,
map_extra: 0,
numa_node: 0,
map_ifindex: 0,
..Default::default()
};
let map = MapHandle::create(MapType::Hash, Some("simple_map"), 8, 64, 1024, &opts).unwrap();
let map_info = MapInfo::new(map.as_fd()).unwrap();
let name_received = map_info.name().unwrap();
assert_eq!(name_received, "simple_map");
assert_eq!(map_info.map_type(), MapType::Hash);
assert_eq!(map_info.flags() & MapFlags::ANY, MapFlags::ANY);
let map_info = &map_info.info;
assert_eq!(map_info.key_size, 8);
assert_eq!(map_info.value_size, 64);
assert_eq!(map_info.max_entries, 1024);
assert_eq!(map_info.btf_id, 0);
assert_eq!(map_info.btf_key_type_id, 0);
assert_eq!(map_info.btf_value_type_id, 0);
assert_eq!(map_info.btf_vmlinux_value_type_id, 0);
assert_eq!(map_info.map_extra, 0);
assert_eq!(map_info.ifindex, 0);
}
#[tag(root)]
#[test]
fn test_object_percpu_lookup() {
let mut obj = get_test_object("percpu_map.bpf.o");
let map = get_map_mut(&mut obj, "percpu_map");
let res = map
.lookup_percpu(&(0_u32).to_ne_bytes(), MapFlags::ANY)
.expect("failed to lookup")
.expect("failed to find value for key");
assert_eq!(
res.len(),
num_possible_cpus().expect("must be one value per cpu")
);
assert_eq!(res[0].len(), size_of::<u32>());
}
#[tag(root)]
#[test]
fn test_object_percpu_invalid_lookup_fn() {
let mut obj = get_test_object("percpu_map.bpf.o");
let map = get_map_mut(&mut obj, "percpu_map");
assert!(map.lookup(&(0_u32).to_ne_bytes(), MapFlags::ANY).is_err());
}
#[tag(root)]
#[test]
fn test_object_percpu_update() {
let mut obj = get_test_object("percpu_map.bpf.o");
let map = get_map_mut(&mut obj, "percpu_map");
let key = (0_u32).to_ne_bytes();
let mut vals: Vec<Vec<u8>> = Vec::new();
for i in 0..num_possible_cpus().unwrap() {
vals.push((i as u32).to_ne_bytes().to_vec());
}
map.update_percpu(&key, &vals, MapFlags::ANY)
.expect("failed to update map");
let res = map
.lookup_percpu(&key, MapFlags::ANY)
.expect("failed to lookup")
.expect("failed to find value for key");
assert_eq!(vals, res);
}
#[tag(root)]
#[test]
fn test_object_percpu_invalid_update_fn() {
let mut obj = get_test_object("percpu_map.bpf.o");
let map = get_map_mut(&mut obj, "percpu_map");
let key = (0_u32).to_ne_bytes();
let val = (1_u32).to_ne_bytes().to_vec();
assert!(map.update(&key, &val, MapFlags::ANY).is_err());
}
#[tag(root)]
#[test]
fn test_object_percpu_lookup_update() {
let mut obj = get_test_object("percpu_map.bpf.o");
let map = get_map_mut(&mut obj, "percpu_map");
let key = (0_u32).to_ne_bytes();
let mut res = map
.lookup_percpu(&key, MapFlags::ANY)
.expect("failed to lookup")
.expect("failed to find value for key");
for e in res.iter_mut() {
e[0] &= 0xf0;
}
map.update_percpu(&key, &res, MapFlags::ANY)
.expect("failed to update after first lookup");
let res2 = map
.lookup_percpu(&key, MapFlags::ANY)
.expect("failed to lookup")
.expect("failed to find value for key");
assert_eq!(res, res2);
}
#[tag(root)]
#[test]
fn test_object_map_empty_lookup() {
let mut obj = get_test_object("runqslower.bpf.o");
let start = get_map_mut(&mut obj, "start");
assert!(start
.lookup(&[1, 2, 3, 4], MapFlags::empty())
.expect("err in map lookup")
.is_none());
}
#[tag(root)]
#[test]
fn test_object_map_queue_crud() {
let mut obj = get_test_object("tracepoint.bpf.o");
let queue = get_map_mut(&mut obj, "queue");
let key: [u8; 0] = [];
let value1 = 42u32.to_ne_bytes();
let value2 = 43u32.to_ne_bytes();
queue
.update(&key, &value1, MapFlags::ANY)
.expect("failed to update in queue");
queue
.update(&key, &value2, MapFlags::ANY)
.expect("failed to update in queue");
let mut val = queue
.lookup(&key, MapFlags::ANY)
.expect("failed to peek the queue")
.expect("failed to retrieve value");
assert_eq!(val.len(), 4);
assert_eq!(&val, &value1);
val = queue
.lookup_and_delete(&key)
.expect("failed to pop from queue")
.expect("failed to retrieve value");
assert_eq!(val.len(), 4);
assert_eq!(&val, &value1);
val = queue
.lookup_and_delete(&key)
.expect("failed to pop from queue")
.expect("failed to retrieve value");
assert_eq!(val.len(), 4);
assert_eq!(&val, &value2);
assert!(queue
.lookup_and_delete(&key)
.expect("failed to pop from queue")
.is_none());
}
#[tag(root)]
#[test]
fn test_object_map_bloom_filter_crud() {
let mut obj = get_test_object("tracepoint.bpf.o");
let bloom_filter = get_map_mut(&mut obj, "bloom_filter");
let key: [u8; 0] = [];
let value1 = 1337u32.to_ne_bytes();
let value2 = 2674u32.to_ne_bytes();
bloom_filter
.update(&key, &value1, MapFlags::ANY)
.expect("failed to add entry value1 to bloom filter");
bloom_filter
.update(&key, &value2, MapFlags::ANY)
.expect("failed to add entry value2 in bloom filter");
bloom_filter
.update(&value1, &value1, MapFlags::ANY)
.expect_err("Non empty key should return an error");
for inserted_value in [value1, value2] {
let val = bloom_filter
.lookup_bloom_filter(&inserted_value)
.expect("failed retrieve item from bloom filter");
assert!(val);
}
let enoent_found = bloom_filter
.lookup_bloom_filter(&[1, 2, 3, 4])
.expect("failed retrieve item from bloom filter");
assert!(!enoent_found);
bloom_filter
.lookup(&[1, 2, 3, 4], MapFlags::ANY)
.expect_err("lookup should fail since we should use lookup_bloom_filter");
bloom_filter
.lookup_and_delete(&key)
.expect_err("Expect delete to fail");
}
#[tag(root)]
#[test]
fn test_object_map_stack_crud() {
let mut obj = get_test_object("tracepoint.bpf.o");
let stack = get_map_mut(&mut obj, "stack");
let key: [u8; 0] = [];
let value1 = 1337u32.to_ne_bytes();
let value2 = 2674u32.to_ne_bytes();
stack
.update(&key, &value1, MapFlags::ANY)
.expect("failed to update in stack");
stack
.update(&key, &value2, MapFlags::ANY)
.expect("failed to update in stack");
let mut val = stack
.lookup(&key, MapFlags::ANY)
.expect("failed to pop from stack")
.expect("failed to retrieve value");
assert_eq!(val.len(), 4);
assert_eq!(&val, &value2);
val = stack
.lookup_and_delete(&key)
.expect("failed to pop from stack")
.expect("failed to retrieve value");
assert_eq!(val.len(), 4);
assert_eq!(&val, &value2);
val = stack
.lookup_and_delete(&key)
.expect("failed to pop from stack")
.expect("failed to retrieve value");
assert_eq!(val.len(), 4);
assert_eq!(&val, &value1);
assert!(stack
.lookup_and_delete(&key)
.expect("failed to pop from stack")
.is_none());
}
#[tag(root)]
#[test]
fn test_object_map_mutation() {
let mut obj = get_test_object("runqslower.bpf.o");
let start = get_map_mut(&mut obj, "start");
start
.update(&[1, 2, 3, 4], &[1, 2, 3, 4, 5, 6, 7, 8], MapFlags::empty())
.expect("failed to write");
let val = start
.lookup(&[1, 2, 3, 4], MapFlags::empty())
.expect("failed to read map")
.expect("failed to find key");
assert_eq!(val.len(), 8);
assert_eq!(val, &[1, 2, 3, 4, 5, 6, 7, 8]);
start.delete(&[1, 2, 3, 4]).expect("failed to delete key");
assert!(start
.lookup(&[1, 2, 3, 4], MapFlags::empty())
.expect("failed to read map")
.is_none());
}
#[tag(root)]
#[test]
fn test_object_map_lookup_into() {
let mut obj = get_test_object("runqslower.bpf.o");
let start = get_map_mut(&mut obj, "start");
start
.update(&[1, 2, 3, 4], &[1, 2, 3, 4, 5, 6, 7, 8], MapFlags::empty())
.expect("failed to write");
let mut value = [0u8; 8];
let found = start
.lookup_into(&[1, 2, 3, 4], &mut value, MapFlags::empty())
.expect("failed to lookup_into");
assert!(found, "key should be found");
assert_eq!(value, [1, 2, 3, 4, 5, 6, 7, 8]);
let mut value2 = [0u8; 8];
let found2 = start
.lookup_into(&[5, 6, 7, 8], &mut value2, MapFlags::empty())
.expect("failed to lookup_into for non-existent key");
assert!(!found2, "key should not be found");
assert_eq!(value2, [0u8; 8]);
}
#[tag(root)]
#[test]
fn test_object_map_lookup_into_wrong_size() {
let mut obj = get_test_object("runqslower.bpf.o");
let start = get_map_mut(&mut obj, "start");
start
.update(&[1, 2, 3, 4], &[1, 2, 3, 4, 5, 6, 7, 8], MapFlags::empty())
.expect("failed to write");
let mut value_small = [0u8; 4];
let result = start.lookup_into(&[1, 2, 3, 4], &mut value_small, MapFlags::empty());
assert!(result.is_err(), "should fail with wrong buffer size");
let mut value_large = [0u8; 16];
let result = start.lookup_into(&[1, 2, 3, 4], &mut value_large, MapFlags::empty());
assert!(result.is_err(), "should fail with wrong buffer size");
}
#[tag(root)]
#[test]
fn test_object_map_lookup_into_consistency() {
let mut obj = get_test_object("runqslower.bpf.o");
let start = get_map_mut(&mut obj, "start");
let test_value = [10, 20, 30, 40, 50, 60, 70, 80];
start
.update(&[1, 2, 3, 4], &test_value, MapFlags::empty())
.expect("failed to write");
let lookup_result = start
.lookup(&[1, 2, 3, 4], MapFlags::empty())
.expect("failed to lookup")
.expect("key not found");
let mut value_buffer = [0u8; 8];
let found = start
.lookup_into(&[1, 2, 3, 4], &mut value_buffer, MapFlags::empty())
.expect("failed to lookup_into");
assert!(found, "key should be found");
assert_eq!(
lookup_result.as_slice(),
&value_buffer,
"lookup() and lookup_into() should return the same value"
);
}
#[tag(root)]
#[test]
fn test_object_map_lookup_flags() {
let mut obj = get_test_object("runqslower.bpf.o");
let start = get_map_mut(&mut obj, "start");
start
.update(&[1, 2, 3, 4], &[1, 2, 3, 4, 5, 6, 7, 8], MapFlags::NO_EXIST)
.expect("failed to write");
assert!(start
.update(&[1, 2, 3, 4], &[1, 2, 3, 4, 5, 6, 7, 8], MapFlags::NO_EXIST)
.is_err());
}
#[tag(root)]
#[test]
fn test_object_map_key_iter() {
let mut obj = get_test_object("runqslower.bpf.o");
let start = get_map_mut(&mut obj, "start");
let key1 = vec![1, 2, 3, 4];
let key2 = vec![1, 2, 3, 5];
let key3 = vec![1, 2, 3, 6];
start
.update(&key1, &[1, 2, 3, 4, 5, 6, 7, 8], MapFlags::empty())
.expect("failed to write");
start
.update(&key2, &[1, 2, 3, 4, 5, 6, 7, 8], MapFlags::empty())
.expect("failed to write");
start
.update(&key3, &[1, 2, 3, 4, 5, 6, 7, 8], MapFlags::empty())
.expect("failed to write");
let mut keys = HashSet::new();
for key in start.keys() {
keys.insert(key);
}
assert_eq!(keys.len(), 3);
assert!(keys.contains(&key1));
assert!(keys.contains(&key2));
assert!(keys.contains(&key3));
}
#[tag(root)]
#[test]
fn test_object_map_key_iter_empty() {
let mut obj = get_test_object("runqslower.bpf.o");
let start = get_map_mut(&mut obj, "start");
let mut count = 0;
for _ in start.keys() {
count += 1;
}
assert_eq!(count, 0);
}
#[tag(root)]
#[test]
fn test_object_map_pin() {
let mut obj = get_test_object("runqslower.bpf.o");
let mut map = get_map_mut(&mut obj, "start");
let path = "/sys/fs/bpf/mymap_test_object_map_pin";
assert!(map.unpin(path).is_err());
assert!(!Path::new(path).exists());
map.pin(path).expect("failed to pin map");
assert!(Path::new(path).exists());
map.unpin(path).expect("failed to unpin map");
assert!(!Path::new(path).exists());
}
#[tag(root)]
#[test]
fn test_object_loading_pinned_map_from_path() {
let mut obj = get_test_object("runqslower.bpf.o");
let mut map = get_map_mut(&mut obj, "start");
let path = "/sys/fs/bpf/mymap_test_pin_to_load_from_path";
map.pin(path).expect("pinning map failed");
let pinned_map = MapHandle::from_pinned_path(path).expect("loading a map from a path failed");
map.unpin(path).expect("unpinning map failed");
assert_eq!(map.name(), pinned_map.name());
assert_eq!(
map.info().unwrap().info.id,
pinned_map.info().unwrap().info.id
);
}
#[tag(root)]
#[test]
fn test_program_loading_fd_from_pinned_path() {
let path = "/sys/fs/bpf/myprog_test_pin_to_load_from_path";
let prog_name = "handle__sched_switch";
let mut obj = get_test_object("runqslower.bpf.o");
let mut prog = get_prog_mut(&mut obj, prog_name);
prog.pin(path).expect("pinning prog failed");
let prog_id = Program::id_from_fd(prog.as_fd()).expect("failed to determine prog id");
let pinned_prog_fd =
Program::fd_from_pinned_path(path).expect("failed to get fd of pinned prog");
let pinned_prog_id =
Program::id_from_fd(pinned_prog_fd.as_fd()).expect("failed to determine pinned prog id");
assert_eq!(prog_id, pinned_prog_id);
prog.unpin(path).expect("unpinning program failed");
}
#[tag(root)]
#[test]
fn test_program_loading_fd_from_pinned_path_with_wrong_pin_type() {
let path = "/sys/fs/bpf/mymap_test_pin_to_load_from_path";
let map_name = "events";
let mut obj = get_test_object("runqslower.bpf.o");
let mut map = get_map_mut(&mut obj, map_name);
map.pin(path).expect("pinning map failed");
let _err = Program::fd_from_pinned_path(path).expect_err("program fd obtained from pinned map");
map.unpin(path).expect("unpinning program failed");
}
#[tag(root)]
#[test]
fn test_object_loading_loaded_map_from_id() {
let mut obj = get_test_object("runqslower.bpf.o");
let map = get_map_mut(&mut obj, "start");
let id = map.info().expect("to get info from map 'start'").info.id;
let map_by_id = MapHandle::from_map_id(id).expect("map to load from id");
assert_eq!(map.name(), map_by_id.name());
assert_eq!(
map.info().unwrap().info.id,
map_by_id.info().unwrap().info.id
);
}
#[tag(root)]
#[test]
fn test_object_programs() {
let mut obj = get_test_object("runqslower.bpf.o");
let _prog = get_prog_mut(&mut obj, "handle__sched_wakeup");
let _prog = get_prog_mut(&mut obj, "handle__sched_wakeup_new");
let _prog = get_prog_mut(&mut obj, "handle__sched_switch");
assert!(!obj.progs().any(|prog| prog.name() == OsStr::new("asdf")));
}
#[tag(root)]
#[test]
fn test_object_programs_iter_mut() {
let obj = get_test_object("runqslower.bpf.o");
assert!(obj.progs().count() == 3);
}
#[tag(root)]
#[test]
fn test_object_program_pin() {
let mut obj = get_test_object("runqslower.bpf.o");
let mut prog = get_prog_mut(&mut obj, "handle__sched_wakeup");
let path = "/sys/fs/bpf/myprog";
assert!(prog.unpin(path).is_err());
assert!(!Path::new(path).exists());
prog.pin(path).expect("failed to pin prog");
assert!(Path::new(path).exists());
defer! {
let _unused = fs::remove_file(path);
}
prog.unpin(path).expect("failed to unpin prog");
assert!(!Path::new(path).exists());
}
#[tag(root)]
#[test]
fn test_object_link_pin() {
let mut obj = get_test_object("runqslower.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__sched_wakeup");
let mut link = prog.attach().expect("failed to attach prog");
let path = "/sys/fs/bpf/mylink";
assert!(link.unpin().is_err());
assert!(!Path::new(path).exists());
link.pin(path).expect("failed to pin prog");
assert!(Path::new(path).exists());
defer! {
let _unused = fs::remove_file(path);
}
link.unpin().expect("failed to unpin prog");
assert!(!Path::new(path).exists());
}
#[tag(root)]
#[test]
fn test_object_reuse_pined_map() {
let path = "/sys/fs/bpf/mymap_test_object_reuse_pined_map";
let key = vec![1, 2, 3, 4];
let val = vec![1, 2, 3, 4, 5, 6, 7, 8];
{
let mut obj = get_test_object("runqslower.bpf.o");
let mut map = get_map_mut(&mut obj, "start");
map.update(&key, &val, MapFlags::empty())
.expect("failed to write");
map.pin(path).expect("failed to pin map");
assert!(Path::new(path).exists());
}
defer! {
let _unused = fs::remove_file(path);
}
let obj_path = get_test_object_path("runqslower.bpf.o");
let mut builder = ObjectBuilder::default();
builder.debug(true);
let mut open_obj = builder.open_file(obj_path).expect("failed to open object");
let mut start = open_obj
.maps_mut()
.find(|map| map.name() == OsStr::new("start"))
.expect("failed to find `start` map");
assert!(start.reuse_pinned_map("/asdf").is_err());
start.reuse_pinned_map(path).expect("failed to reuse map");
let mut obj = open_obj.load().expect("failed to load object");
let mut reused_map = get_map_mut(&mut obj, "start");
let found_val = reused_map
.lookup(&key, MapFlags::empty())
.expect("failed to read map")
.expect("failed to find key");
assert_eq!(&found_val, &val);
reused_map.unpin(path).expect("failed to unpin map");
assert!(!Path::new(path).exists());
}
#[tag(root)]
#[test]
fn test_object_ringbuf_raw() {
let mut obj = get_test_object("ringbuf.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__sys_enter_getpid");
let _link = prog.attach().expect("failed to attach prog");
static V1: AtomicI32 = AtomicI32::new(0);
static V2: AtomicI32 = AtomicI32::new(0);
fn callback1(data: &[u8]) -> i32 {
let mut value: i32 = 0;
plain::copy_from_bytes(&mut value, data).expect("Wrong size");
V1.store(value, Ordering::SeqCst);
0
}
fn callback2(data: &[u8]) -> i32 {
let mut value: i32 = 0;
plain::copy_from_bytes(&mut value, data).expect("Wrong size");
V2.store(value, Ordering::SeqCst);
0
}
let builder = libbpf_rs::RingBufferBuilder::new();
assert!(
builder.build().is_err(),
"Should not be able to build without adding at least one ringbuf"
);
let mut builder = libbpf_rs::RingBufferBuilder::new();
let map1 = get_map(&obj, "ringbuf1");
builder
.add(&map1, callback1)
.expect("failed to add ringbuf");
let map2 = get_map(&obj, "ringbuf2");
builder
.add(&map2, callback2)
.expect("failed to add ringbuf");
let mgr = builder.build().expect("failed to build");
unsafe { libc::getpid() };
let ret = mgr.consume_raw();
assert!(ret >= 2);
assert_eq!(V1.load(Ordering::SeqCst), 1);
assert_eq!(V2.load(Ordering::SeqCst), 2);
let ret = mgr.consume_raw();
assert!(ret >= 0);
let ret = mgr.poll_raw(Duration::from_millis(100));
assert!(ret >= 0);
for _ in 1..=10 {
unsafe { libc::getpid() };
}
let ret = mgr.consume_raw_n(1);
assert!(ret == 1);
let ret = mgr.consume_raw_n(2);
assert!(ret == 2);
let ret = mgr.consume_raw_n(10);
assert!((7..=10).contains(&ret));
}
#[tag(root)]
#[test]
fn test_object_ringbuf_err_callback() {
let mut obj = get_test_object("ringbuf.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__sys_enter_getpid");
let _link = prog.attach().expect("failed to attach prog");
fn callback1(_data: &[u8]) -> i32 {
-libc::ENOENT
}
fn callback2(_data: &[u8]) -> i32 {
-libc::EPERM
}
let builder = libbpf_rs::RingBufferBuilder::new();
assert!(
builder.build().is_err(),
"Should not be able to build without adding at least one ringbuf"
);
let mut builder = libbpf_rs::RingBufferBuilder::new();
let map1 = get_map(&obj, "ringbuf1");
builder
.add(&map1, callback1)
.expect("failed to add ringbuf");
let map2 = get_map(&obj, "ringbuf2");
builder
.add(&map2, callback2)
.expect("failed to add ringbuf");
let mgr = builder.build().expect("failed to build");
unsafe { libc::getpid() };
let ret = mgr.consume_raw();
assert!(ret == -libc::ENOENT || ret == -libc::EPERM);
unsafe { libc::getpid() };
let ret = mgr.poll_raw(Duration::from_millis(100));
assert!(ret == -libc::ENOENT || ret == -libc::EPERM);
}
#[tag(root)]
#[test]
fn test_object_ringbuf() {
let mut obj = get_test_object("ringbuf.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__sys_enter_getpid");
let _link = prog.attach().expect("failed to attach prog");
static V1: AtomicI32 = AtomicI32::new(0);
static V2: AtomicI32 = AtomicI32::new(0);
fn callback1(data: &[u8]) -> i32 {
let mut value: i32 = 0;
plain::copy_from_bytes(&mut value, data).expect("Wrong size");
V1.store(value, Ordering::SeqCst);
0
}
fn callback2(data: &[u8]) -> i32 {
let mut value: i32 = 0;
plain::copy_from_bytes(&mut value, data).expect("Wrong size");
V2.store(value, Ordering::SeqCst);
0
}
let builder = libbpf_rs::RingBufferBuilder::new();
assert!(
builder.build().is_err(),
"Should not be able to build without adding at least one ringbuf"
);
let mut builder = libbpf_rs::RingBufferBuilder::new();
let map1 = get_map(&obj, "ringbuf1");
builder
.add(&map1, callback1)
.expect("failed to add ringbuf");
let map2 = get_map(&obj, "ringbuf2");
builder
.add(&map2, callback2)
.expect("failed to add ringbuf");
let mgr = builder.build().expect("failed to build");
unsafe { libc::getpid() };
mgr.consume().expect("failed to consume ringbuf");
assert_eq!(V1.load(Ordering::SeqCst), 1);
assert_eq!(V2.load(Ordering::SeqCst), 2);
V1.store(0, Ordering::SeqCst);
V2.store(0, Ordering::SeqCst);
unsafe { libc::getpid() };
mgr.poll(Duration::from_millis(100))
.expect("failed to poll ringbuf");
assert_eq!(V1.load(Ordering::SeqCst), 1);
assert_eq!(V2.load(Ordering::SeqCst), 2);
}
#[tag(root)]
#[test]
fn test_object_ringbuf_closure() {
let mut obj = get_test_object("ringbuf.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__sys_enter_getpid");
let _link = prog.attach().expect("failed to attach prog");
let (sender1, receiver1) = channel();
let callback1 = move |data: &[u8]| -> i32 {
let mut value: i32 = 0;
plain::copy_from_bytes(&mut value, data).expect("Wrong size");
sender1.send(value).expect("failed to send value");
0
};
let (sender2, receiver2) = channel();
let callback2 = move |data: &[u8]| -> i32 {
let mut value: i32 = 0;
plain::copy_from_bytes(&mut value, data).expect("Wrong size");
sender2.send(value).expect("failed to send value");
0
};
let builder = libbpf_rs::RingBufferBuilder::new();
assert!(
builder.build().is_err(),
"Should not be able to build without adding at least one ringbuf"
);
let mut builder = libbpf_rs::RingBufferBuilder::new();
let map1 = get_map(&obj, "ringbuf1");
builder
.add(&map1, callback1)
.expect("failed to add ringbuf");
let map2 = get_map(&obj, "ringbuf2");
builder
.add(&map2, callback2)
.expect("failed to add ringbuf");
let mgr = builder.build().expect("failed to build");
unsafe { libc::getpid() };
mgr.consume().expect("failed to consume ringbuf");
let v1 = receiver1.recv().expect("failed to receive value");
let v2 = receiver2.recv().expect("failed to receive value");
assert_eq!(v1, 1);
assert_eq!(v2, 2);
}
#[tag(root)]
#[test]
fn test_object_ringbuf_with_closed_map() {
fn test(poll_fn: impl FnOnce(&libbpf_rs::RingBuffer)) {
let mut value = 0i32;
{
let mut obj = get_test_object("tracepoint.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__tracepoint");
let _link = prog
.attach_tracepoint(TracepointCategory::Syscalls, "sys_enter_getpid")
.expect("failed to attach prog");
let map = get_map_mut(&mut obj, "ringbuf");
let callback = |data: &[u8]| {
plain::copy_from_bytes(&mut value, data).expect("Wrong size");
0
};
let mut builder = libbpf_rs::RingBufferBuilder::new();
builder.add(&map, callback).expect("failed to add ringbuf");
let ringbuf = builder.build().expect("failed to build");
drop(obj);
let _pid = unsafe { libc::getpid() };
let () = poll_fn(&ringbuf);
}
assert_eq!(value, 1);
}
test(|ringbuf| ringbuf.consume().expect("failed to consume ringbuf"));
test(|ringbuf| {
ringbuf
.poll(Duration::from_secs(5))
.expect("failed to poll ringbuf")
});
}
#[tag(root)]
#[test]
fn test_object_user_ringbuf() {
#[repr(C)]
struct MyStruct {
key: u32,
value: u32,
}
unsafe impl Plain for MyStruct {}
let mut obj = get_test_object("user_ringbuf.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__sys_enter_getpid");
let _link = prog.attach().expect("failed to attach prog");
let urb_map = get_map_mut(&mut obj, "user_ringbuf");
let user_ringbuf = UserRingBuffer::new(&urb_map).expect("failed to create user ringbuf");
let mut urb_sample = user_ringbuf
.reserve(size_of::<MyStruct>())
.expect("failed to reserve space");
let bytes = urb_sample.as_mut();
let my_struct = plain::from_mut_bytes::<MyStruct>(bytes).expect("failed to convert bytes");
my_struct.key = 42;
my_struct.value = 1337;
user_ringbuf
.submit(urb_sample)
.expect("failed to submit sample");
let _pid = unsafe { libc::getpid() };
let samples_map = get_map_mut(&mut obj, "samples");
let key: u32 = 42;
let value: u32 = 1337;
let res = samples_map
.lookup(&key.to_ne_bytes(), MapFlags::ANY)
.expect("failed to lookup")
.expect("failed to find value for key");
assert_eq!(res.len(), size_of::<u32>());
let mut array = [0; size_of::<u32>()];
array.copy_from_slice(&res[..]);
assert_eq!(u32::from_ne_bytes(array), value);
}
#[tag(root)]
#[test]
fn test_object_user_ringbuf_reservation_too_big() {
let mut obj = get_test_object("user_ringbuf.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__sys_enter_getpid");
let _link = prog.attach().expect("failed to attach prog");
let urb_map = get_map_mut(&mut obj, "user_ringbuf");
let user_ringbuf = UserRingBuffer::new(&urb_map).expect("failed to create user ringbuf");
let err = user_ringbuf.reserve(1024 * 1024).unwrap_err();
assert!(
err.to_string().contains("requested size is too large"),
"{err:#}"
);
}
#[tag(root)]
#[test]
fn test_object_user_ringbuf_not_enough_space() {
let mut obj = get_test_object("user_ringbuf.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__sys_enter_getpid");
let _link = prog.attach().expect("failed to attach prog");
let urb_map = get_map_mut(&mut obj, "user_ringbuf");
let user_ringbuf = UserRingBuffer::new(&urb_map).expect("failed to create user ringbuf");
let _sample = user_ringbuf
.reserve(1024 * 3)
.expect("failed to reserve space");
let err = user_ringbuf.reserve(1024 * 3).unwrap_err();
assert!(
err.to_string()
.contains("not enough space in the ring buffer"),
"{err:#}"
);
}
#[tag(root)]
#[test]
fn test_object_task_iter() {
#[repr(C)]
#[derive(Clone, Copy)]
struct IndexPidPair {
i: u32,
pid: i32,
}
unsafe impl Plain for IndexPidPair {}
fn test_iter(link: libbpf_rs::Link) {
let mut iter = Iter::new(&link).expect("failed to create iterator");
let mut buf = Vec::new();
let bytes_read = iter
.read_to_end(&mut buf)
.expect("failed to read from iterator");
assert!(bytes_read > 0);
assert_eq!(bytes_read % size_of::<IndexPidPair>(), 0);
let items: &[IndexPidPair] =
plain::slice_from_bytes(buf.as_slice()).expect("Input slice cannot satisfy length");
assert!(!items.is_empty());
assert_eq!(items[0].i, 0);
assert!(items.windows(2).all(|w| w[0].i + 1 == w[1].i));
assert!(items.iter().any(|&item| item.pid == 1));
}
let mut obj = get_test_object("taskiter.bpf.o");
let prog = get_prog_mut(&mut obj, "dump_pid");
let link_autoattach = prog.attach().expect("failed to auto-attach prog");
test_iter(link_autoattach);
let mut obj = get_test_object("taskiter.bpf.o");
let prog = get_prog_mut(&mut obj, "dump_pid");
let link_noopts = prog
.attach_iter_with_opts(libbpf_rs::IterOpts::None)
.expect("failed to attach prog with no opts");
test_iter(link_noopts);
}
#[tag(root)]
#[test]
fn test_object_map_iter() {
let opts = libbpf_sys::bpf_map_create_opts {
sz: size_of::<libbpf_sys::bpf_map_create_opts>() as libbpf_sys::size_t,
map_flags: libbpf_sys::BPF_F_NO_PREALLOC,
..Default::default()
};
let map = MapHandle::create(
MapType::Hash,
Some("mymap_test_object_map_iter"),
4,
8,
8,
&opts,
)
.expect("failed to create map");
for i in 0..3 {
let key = i32::to_ne_bytes(i);
let val = [&key[..], &[0_u8; 4]].concat();
map.update(&key, val.as_slice(), MapFlags::empty())
.expect("failed to write");
}
let mut obj = get_test_object("mapiter.bpf.o");
let prog = get_prog_mut(&mut obj, "map_iter");
let link = prog
.attach_iter(map.as_fd())
.expect("failed to attach map iter prog");
let mut iter = Iter::new(&link).expect("failed to create map iterator");
let mut buf = Vec::new();
let bytes_read = iter
.read_to_end(&mut buf)
.expect("failed to read from iterator");
assert!(bytes_read > 0);
assert_eq!(bytes_read % size_of::<u32>(), 0);
let buf =
plain::slice_from_bytes::<u32>(buf.as_slice()).expect("Input slice cannot satisfy length");
assert!(buf.contains(&0));
assert!(buf.contains(&1));
assert!(buf.contains(&2));
}
#[tag(root)]
#[test]
fn test_object_map_create_and_pin() {
let opts = libbpf_sys::bpf_map_create_opts {
sz: size_of::<libbpf_sys::bpf_map_create_opts>() as libbpf_sys::size_t,
map_flags: libbpf_sys::BPF_F_NO_PREALLOC,
..Default::default()
};
let mut map = MapHandle::create(
MapType::Hash,
Some("mymap_test_object_map_create_and_pin"),
4,
8,
8,
&opts,
)
.expect("failed to create map");
assert_eq!(map.name(), "mymap_test_object_map_create_and_pin");
let key = vec![1, 2, 3, 4];
let val = vec![1, 2, 3, 4, 5, 6, 7, 8];
map.update(&key, &val, MapFlags::empty())
.expect("failed to write");
let res = map
.lookup(&key, MapFlags::ANY)
.expect("failed to lookup")
.expect("failed to find value for key");
assert_eq!(val, res);
let path = "/sys/fs/bpf/mymap_test_object_map_create_and_pin";
assert!(map.unpin(path).is_err());
assert!(!Path::new(path).exists());
map.pin(path).expect("failed to pin map");
assert!(Path::new(path).exists());
map.unpin(path).expect("failed to unpin map");
assert!(!Path::new(path).exists());
}
#[tag(root)]
#[test]
fn test_object_map_create_without_name() {
let opts = libbpf_sys::bpf_map_create_opts {
sz: size_of::<libbpf_sys::bpf_map_create_opts>() as libbpf_sys::size_t,
map_flags: libbpf_sys::BPF_F_NO_PREALLOC,
btf_fd: 0,
btf_key_type_id: 0,
btf_value_type_id: 0,
btf_vmlinux_value_type_id: 0,
inner_map_fd: 0,
map_extra: 0,
numa_node: 0,
map_ifindex: 0,
..Default::default()
};
let map = MapHandle::create(MapType::Hash, Option::<&str>::None, 4, 8, 8, &opts)
.expect("failed to create map");
assert!(map.name().is_empty());
let key = vec![1, 2, 3, 4];
let val = vec![1, 2, 3, 4, 5, 6, 7, 8];
map.update(&key, &val, MapFlags::empty())
.expect("failed to write");
let res = map
.lookup(&key, MapFlags::ANY)
.expect("failed to lookup")
.expect("failed to find value for key");
assert_eq!(val, res);
}
#[tag(root)]
#[test]
fn test_object_map_handle_clone() {
let mut obj = get_test_object("runqslower.bpf.o");
let map = get_map_mut(&mut obj, "events");
let handle1 = MapHandle::try_from(&map).expect("failed to create handle from Map");
assert_eq!(map.name(), handle1.name());
assert_eq!(map.map_type(), handle1.map_type());
assert_eq!(map.key_size(), handle1.key_size());
assert_eq!(map.value_size(), handle1.value_size());
assert_eq!(map.max_entries(), handle1.max_entries());
let handle2 = MapHandle::try_from(&handle1).expect("failed to duplicate existing handle");
assert_eq!(handle1.name(), handle2.name());
assert_eq!(handle1.map_type(), handle2.map_type());
assert_eq!(handle1.key_size(), handle2.key_size());
assert_eq!(handle1.value_size(), handle2.value_size());
assert_eq!(handle1.max_entries(), handle2.max_entries());
let info1 = map.info().expect("failed to get map info from map");
let info2 = handle2.info().expect("failed to get map info from handle");
assert_eq!(
info1.info.id, info2.info.id,
"Map and MapHandle have different IDs"
);
}
#[tag(root)]
#[test]
fn test_object_usdt() {
let mut obj = get_test_object("usdt.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__usdt");
let path = current_exe().expect("failed to find executable name");
let _link = prog
.attach_usdt(
unsafe { libc::getpid() },
&path,
"test_provider",
"test_function",
)
.expect("failed to attach prog");
let map = get_map_mut(&mut obj, "ringbuf");
let action = || {
probe!(test_provider, test_function, 1);
};
let result = with_ringbuffer(&map, action);
assert_eq!(result, 1);
}
#[tag(root)]
#[test]
fn test_object_usdt_cookie() {
let cookie_val = 1337u16;
let mut obj = get_test_object("usdt.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__usdt_with_cookie");
let path = current_exe().expect("failed to find executable name");
let _link = prog
.attach_usdt_with_opts(
unsafe { libc::getpid() },
&path,
"test_provider",
"test_function2",
UsdtOpts {
cookie: cookie_val.into(),
..UsdtOpts::default()
},
)
.expect("failed to attach prog");
let map = get_map_mut(&mut obj, "ringbuf");
let action = || {
probe!(test_provider, test_function2, 1);
};
let result = with_ringbuffer(&map, action);
assert_eq!(result, cookie_val.into());
}
#[tag(root)]
#[test]
fn test_map_probes() {
let supported = MapType::Array
.is_supported()
.expect("failed to query if Array map is supported");
assert!(supported);
let supported_res = MapType::Unknown.is_supported();
assert!(supported_res.is_err());
}
#[tag(root)]
#[test]
fn test_program_probes() {
let supported = ProgramType::SocketFilter
.is_supported()
.expect("failed to query if SocketFilter program is supported");
assert!(supported);
let supported_res = ProgramType::Unknown.is_supported();
assert!(supported_res.is_err());
}
#[tag(root)]
#[test]
fn test_program_helper_probes() {
let supported = ProgramType::SocketFilter
.is_helper_supported(libbpf_sys::BPF_FUNC_map_lookup_elem)
.expect("failed to query if helper supported");
assert!(supported);
let supported = ProgramType::SocketFilter
.is_helper_supported(libbpf_sys::BPF_FUNC_redirect)
.expect("failed to query if helper supported");
assert!(!supported);
let supported_res = MapType::Unknown.is_supported();
assert!(supported_res.is_err());
}
#[tag(root)]
#[test]
fn test_object_open_program_insns() {
let open_obj = open_test_object("usdt.bpf.o");
let prog = open_obj
.progs()
.find(|prog| prog.name() == OsStr::new("handle__usdt"))
.expect("failed to find program");
let insns = prog.insns();
assert!(!insns.is_empty());
}
#[tag(root)]
#[test]
fn test_object_program_insns() {
let mut obj = get_test_object("usdt.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__usdt");
let insns = prog.insns();
assert!(!insns.is_empty());
}
#[tag(root)]
#[test]
fn test_object_program_autoload() {
let mut open_obj = open_test_object("kprobe.bpf.o");
let prog_name = "handle__kprobe";
let mut open_prog = open_obj
.progs_mut()
.find(|prog| prog.name() == prog_name)
.expect("failed to find `handle__kprobe` program");
assert!(open_prog.autoload());
open_prog.set_autoload(false);
assert!(!open_prog.autoload());
let mut obj = open_obj.load().expect("failed to load object");
let prog = get_prog_mut(&mut obj, prog_name);
assert!(!prog.autoload());
assert!(prog.as_fd().as_raw_fd() < 0); }
#[tag(root)]
#[test]
fn test_object_kprobe() {
let mut obj = get_test_object("kprobe.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__kprobe");
let _link = prog
.attach_kprobe(false, "bpf_fentry_test1")
.expect("failed to attach prog");
}
#[tag(root)]
#[test]
fn test_object_kprobe_with_opts() {
let mut obj = get_test_object("kprobe.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__kprobe");
let opts = KprobeOpts::default();
let _link = prog
.attach_kprobe_with_opts(false, "bpf_fentry_test1", opts)
.expect("failed to attach prog");
}
#[tag(root)]
#[test]
#[ignore = "requires kernel with kprobe multi support"]
fn test_object_kprobe_multi() {
let mut open_obj = open_test_object("kprobe.bpf.o");
open_obj
.progs_mut()
.find(|prog| prog.name() == "handle__kprobe")
.expect("failed to find `handle__kprobe` program")
.set_attach_type(libbpf_rs::ProgramAttachType::KprobeMulti);
let mut obj = open_obj.load().expect("failed to load object");
let prog = get_prog_mut(&mut obj, "handle__kprobe");
let _link = prog
.attach_kprobe_multi(false, vec!["bpf_fentry_test1", "bpf_fentry_test2"])
.expect("failed to attach prog");
}
#[tag(root)]
#[test]
#[ignore = "requires kernel with kprobe multi support"]
fn test_object_kprobe_multi_with_opts() {
let mut open_obj = open_test_object("kprobe.bpf.o");
open_obj
.progs_mut()
.find(|prog| prog.name() == "handle__kprobe")
.expect("failed to find `handle__kprobe` program")
.set_attach_type(libbpf_rs::ProgramAttachType::KprobeMulti);
let mut obj = open_obj.load().expect("failed to load object");
let prog = get_prog_mut(&mut obj, "handle__kprobe");
let opts = KprobeMultiOpts {
symbols: vec![
"bpf_fentry_test1".to_string(),
"bpf_fentry_test2".to_string(),
],
..Default::default()
};
let _link = prog
.attach_kprobe_multi_with_opts(opts)
.expect("failed to attach prog");
}
#[tag(root)]
#[test]
fn test_object_tracepoint() {
let mut obj = get_test_object("tracepoint.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__tracepoint");
let _link = prog
.attach_tracepoint(TracepointCategory::Syscalls, "sys_enter_getpid")
.expect("failed to attach prog");
let map = get_map_mut(&mut obj, "ringbuf");
let action = || {
let _pid = unsafe { libc::getpid() };
};
let result = with_ringbuffer(&map, action);
assert_eq!(result, 1);
}
#[tag(root)]
#[test]
fn test_object_tracepoint_with_opts() {
let cookie_val = 42u16;
let mut obj = get_test_object("tracepoint.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__tracepoint_with_cookie");
let opts = TracepointOpts {
cookie: cookie_val.into(),
..TracepointOpts::default()
};
let _link = prog
.attach_tracepoint_with_opts(TracepointCategory::Syscalls, "sys_enter_getpid", opts)
.expect("failed to attach prog");
let map = get_map_mut(&mut obj, "ringbuf");
let action = || {
let _pid = unsafe { libc::getpid() };
};
let result = with_ringbuffer(&map, action);
assert_eq!(result, cookie_val.into());
}
#[tag(root)]
#[test]
fn test_object_raw_tracepoint() {
let mut open_obj = open_test_object("tracepoint.bpf.o");
open_obj
.progs_mut()
.find(|prog| prog.name() == "handle__tracepoint")
.expect("failed to find `handle__tracepoint` program")
.set_prog_type(libbpf_rs::ProgramType::RawTracepoint);
let mut obj = open_obj.load().expect("failed to load object");
let prog = get_prog_mut(&mut obj, "handle__tracepoint");
let _link = prog
.attach_raw_tracepoint("sys_enter")
.expect("failed to attach prog");
let map = get_map_mut(&mut obj, "ringbuf");
let action = || {
let _pid = unsafe { libc::getpid() };
};
let result = with_ringbuffer(&map, action);
assert_eq!(result, 1);
}
#[tag(root)]
#[test]
#[ignore = "requires kernel with bpf_get_attach_cookie for raw tracepoints"]
fn test_object_raw_tracepoint_with_opts() {
let cookie_val = 42u16;
let mut open_obj = open_test_object("tracepoint.bpf.o");
open_obj
.progs_mut()
.find(|prog| prog.name() == "handle__tracepoint_with_cookie")
.expect("failed to find `handle__tracepoint` program")
.set_prog_type(libbpf_rs::ProgramType::RawTracepoint);
let mut obj = open_obj.load().expect("failed to load object");
let prog = get_prog_mut(&mut obj, "handle__tracepoint_with_cookie");
let opts = RawTracepointOpts {
cookie: cookie_val.into(),
..Default::default()
};
let _link = prog
.attach_raw_tracepoint_with_opts("sys_enter", opts)
.expect("failed to attach prog");
let map = get_map_mut(&mut obj, "ringbuf");
let action = || {
let _pid = unsafe { libc::getpid() };
};
let result = with_ringbuffer(&map, action);
assert_eq!(result, cookie_val.into());
}
#[inline(never)]
#[no_mangle]
extern "C" fn uprobe_multi_func_1() -> usize {
hint::black_box(42)
}
#[inline(never)]
#[no_mangle]
extern "C" fn uprobe_multi_func_2() -> usize {
hint::black_box(43)
}
#[inline(never)]
#[no_mangle]
extern "C" fn multi_uprobe_func_with_opts_func_1() -> usize {
hint::black_box(44)
}
#[inline(never)]
#[no_mangle]
extern "C" fn multi_uprobe_func_with_opts_func_2() -> usize {
hint::black_box(45)
}
#[inline(never)]
#[no_mangle]
extern "C" fn non_default_opts_multi_uprobe_func_with_opts_func_1() -> usize {
hint::black_box(46)
}
#[inline(never)]
#[no_mangle]
extern "C" fn non_default_opts_multi_uprobe_func_with_opts_func_2() -> usize {
hint::black_box(47)
}
#[tag(root)]
#[test]
fn test_object_uprobe_multi_with_opts() {
let mut obj = get_test_object("uprobe_multi.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__uprobe_multi_with_opts");
let func_pattern = "multi_uprobe_func_*";
let pid = unsafe { libc::getpid() };
let path = current_exe().expect("failed to find executable name");
let opts = UprobeMultiOpts::default();
let _link = prog
.attach_uprobe_multi_with_opts(pid, path, func_pattern, opts)
.expect("failed to attach uprobe multi");
multi_uprobe_func_with_opts_func_1();
multi_uprobe_func_with_opts_func_2();
let map = get_map_mut(&mut obj, "hash_map");
let result_bytes = map
.lookup(&(1_u32).to_ne_bytes(), MapFlags::ANY)
.expect("failed to lookup")
.expect("failed to find value for key");
let result = i32::from_ne_bytes(
result_bytes
.as_slice()
.try_into()
.expect("invalid value size"),
);
assert_eq!(result, 2);
}
#[tag(root)]
#[test]
fn test_object_uprobe_multi_with_non_default_opts() {
let mut obj = get_test_object("uprobe_multi.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__uprobe_multi_with_non_default_opts");
let func_pattern = "";
let pid = unsafe { libc::getpid() };
let path = current_exe().expect("failed to find executable name");
let opts = UprobeMultiOpts {
syms: vec![
"non_default_opts_multi_uprobe_func_with_opts_func_1".to_string(),
"non_default_opts_multi_uprobe_func_with_opts_func_2".to_string(),
],
..Default::default()
};
let _link = prog
.attach_uprobe_multi_with_opts(pid, path, func_pattern, opts)
.expect("failed to attach uprobe multi");
non_default_opts_multi_uprobe_func_with_opts_func_1();
non_default_opts_multi_uprobe_func_with_opts_func_2();
let map = get_map_mut(&mut obj, "hash_map");
let result_bytes = map
.lookup(&(1_u32).to_ne_bytes(), MapFlags::ANY)
.expect("failed to lookup")
.expect("failed to find value for key");
let result = i32::from_ne_bytes(
result_bytes
.as_slice()
.try_into()
.expect("invalid value size"),
);
assert_eq!(result, 2);
}
#[tag(root)]
#[test]
fn test_object_uprobe_multi() {
let mut obj = get_test_object("uprobe_multi.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__uprobe_multi");
let func_pattern = "uprobe_multi_func_*";
let pid = unsafe { libc::getpid() };
let path = current_exe().expect("failed to find executable name");
let _link = prog
.attach_uprobe_multi(pid, path, func_pattern, false, false)
.expect("failed to attach uprobe multi");
uprobe_multi_func_1();
uprobe_multi_func_2();
let map = get_map_mut(&mut obj, "hash_map");
let result_bytes = map
.lookup(&(0_u32).to_ne_bytes(), MapFlags::ANY)
.expect("failed to lookup")
.expect("failed to find value for key");
let result = i32::from_ne_bytes(
result_bytes
.as_slice()
.try_into()
.expect("invalid value size"),
);
assert_eq!(result, 2);
}
#[inline(never)]
#[no_mangle]
extern "C" fn uprobe_target() -> usize {
hint::black_box(42)
}
#[tag(root)]
#[test]
fn test_object_uprobe_with_opts() {
let mut obj = get_test_object("uprobe.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__uprobe");
let pid = unsafe { libc::getpid() };
let path = current_exe().expect("failed to find executable name");
let func_offset = 0;
let opts = UprobeOpts {
func_name: Some("uprobe_target".into()),
..Default::default()
};
let _link = prog
.attach_uprobe_with_opts(pid, path, func_offset, opts)
.expect("failed to attach prog");
let map = get_map_mut(&mut obj, "ringbuf");
let action = || {
let _ = uprobe_target();
};
let result = with_ringbuffer(&map, action);
assert_eq!(result, 1);
}
#[tag(root)]
#[test]
fn test_object_uprobe_with_func_offset() {
let mut obj = get_test_object("uprobe.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__uprobe");
let pid = unsafe { libc::getpid() };
let path = current_exe().expect("failed to find executable name");
let func_offset = get_symbol_offset(&path, "uprobe_target").unwrap();
let _link = prog
.attach_uprobe_with_opts(pid, path, func_offset, Default::default())
.expect("failed to attach prog");
let map = get_map_mut(&mut obj, "ringbuf");
let action = || {
let _ = uprobe_target();
};
let result = with_ringbuffer(&map, action);
assert_eq!(result, 1);
}
#[tag(root)]
#[test]
fn test_object_uprobe_with_cookie() {
let cookie_val = 5u16;
let mut obj = get_test_object("uprobe.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__uprobe_with_cookie");
let pid = unsafe { libc::getpid() };
let path = current_exe().expect("failed to find executable name");
let func_offset = 0;
let opts = UprobeOpts {
func_name: Some("uprobe_target".into()),
cookie: cookie_val.into(),
..Default::default()
};
let _link = prog
.attach_uprobe_with_opts(pid, path, func_offset, opts)
.expect("failed to attach prog");
let map = get_map_mut(&mut obj, "ringbuf");
let action = || {
let _ = uprobe_target();
};
let result = with_ringbuffer(&map, action);
assert_eq!(result, cookie_val.into());
}
#[test]
fn test_object_link_files_buffers() {
fn test(files: Vec<&PathBuf>, buffers_files: Vec<&PathBuf>) {
let output_file = NamedTempFile::new().unwrap();
let mut linker = Linker::new(output_file.path()).unwrap();
let () = files
.into_iter()
.try_for_each(|file| linker.add_file(file))
.unwrap();
let buffers: Vec<Vec<u8>> = buffers_files
.into_iter()
.map(|path| fs::read(path).expect("failed to read object file"))
.collect();
let () = buffers
.iter()
.try_for_each(|buf| linker.add_buf(buf))
.unwrap();
let () = linker.link().unwrap();
let _object = ObjectBuilder::default()
.debug(true)
.open_file(output_file.path())
.unwrap();
}
let obj_path1 = get_test_object_path("usdt.bpf.o");
let obj_path2 = get_test_object_path("ringbuf.bpf.o");
test(vec![&obj_path1], vec![]);
test(vec![&obj_path1, &obj_path2], vec![]);
test(vec![], vec![&obj_path1]);
test(vec![], vec![&obj_path1, &obj_path2]);
test(vec![&obj_path1], vec![&obj_path2]);
}
#[tag(root)]
#[test]
fn test_perf_event_link_info_tracepoint() {
let mut tp_obj = get_test_object("tracepoint.bpf.o");
let tp_prog = get_prog_mut(&mut tp_obj, "handle__tracepoint");
let tp_link = tp_prog
.attach_tracepoint(TracepointCategory::Syscalls, "sys_enter_getpid")
.expect("failed to attach tracepoint");
let tp_info = tp_link.info().expect("failed to get tracepoint link info");
let LinkTypeInfo::PerfEvent(perf_info) = &tp_info.info else {
panic!(
"Expected LinkTypeInfo::PerfEvent for tracepoint, got: {:?}",
tp_info.info
);
};
let PerfEventType::Tracepoint { name, .. } = &perf_info.event_type else {
panic!(
"Expected PerfEventType::Tracepoint, got: {:?}",
perf_info.event_type
);
};
let tp_name = name.as_ref().expect("tracepoint should have a name");
assert!(*tp_name == CString::new("sys_enter_getpid").unwrap());
}
#[tag(root)]
#[test]
fn test_perf_event_link_info_kprobe() {
let mut kprobe_obj = get_test_object("kprobe.bpf.o");
let kprobe_prog = get_prog_mut(&mut kprobe_obj, "handle__kprobe");
let kprobe_link = kprobe_prog
.attach_kprobe(false, "bpf_fentry_test1")
.expect("failed to attach kprobe");
let kprobe_info = kprobe_link.info().expect("failed to get kprobe link info");
let LinkTypeInfo::PerfEvent(perf_info) = &kprobe_info.info else {
panic!(
"Expected LinkTypeInfo::PerfEvent for kprobe, got: {:?}",
kprobe_info.info
);
};
let PerfEventType::Kprobe {
func_name,
is_retprobe,
..
} = &perf_info.event_type
else {
panic!(
"Expected PerfEventType::Kprobe, got: {:?}",
perf_info.event_type
);
};
assert!(!is_retprobe, "Expected kprobe (not retprobe)");
let name = func_name
.as_ref()
.expect("kprobe should have a function name");
assert_eq!(*name, CString::new("bpf_fentry_test1").unwrap());
}
#[tag(root)]
#[test]
fn test_perf_event_link_info_kretprobe() {
let mut kretprobe_obj = get_test_object("kprobe.bpf.o");
let kretprobe_prog = get_prog_mut(&mut kretprobe_obj, "handle__kprobe");
let kretprobe_link = kretprobe_prog
.attach_kprobe(true, "bpf_fentry_test1")
.expect("failed to attach kretprobe");
let kretprobe_info = kretprobe_link
.info()
.expect("failed to get kretprobe link info");
let LinkTypeInfo::PerfEvent(perf_info) = &kretprobe_info.info else {
panic!(
"Expected LinkTypeInfo::PerfEvent for kretprobe, got: {:?}",
kretprobe_info.info
);
};
let PerfEventType::Kprobe {
func_name,
is_retprobe,
..
} = &perf_info.event_type
else {
panic!(
"Expected PerfEventType::Kprobe, got: {:?}",
perf_info.event_type
);
};
assert!(*is_retprobe, "Expected kretprobe");
let name = func_name
.as_ref()
.expect("kretprobe should have a function name");
assert_eq!(*name, CString::new("bpf_fentry_test1").unwrap());
}
fn attach_uprobe_get_info(
prog: &libbpf_rs::ProgramMut,
path: &PathBuf,
offset: usize,
opts: &UprobeOpts,
) -> (Option<CString>, bool, u32, u64, u64) {
let pid = unsafe { libc::getpid() };
let link = prog
.attach_uprobe_with_opts(pid, path, offset, opts.clone())
.expect("failed to attach uprobe");
let link_info = link.info().expect("failed to get uprobe link info");
let LinkTypeInfo::PerfEvent(perf_info) = link_info.info else {
panic!(
"Expected LinkTypeInfo::PerfEvent for uprobe, got: {:?}",
link_info.info
);
};
let PerfEventType::Uprobe {
file_name,
is_retprobe,
offset,
cookie,
ref_ctr_offset,
} = perf_info.event_type
else {
panic!(
"Expected PerfEventType::Uprobe, got: {:?}",
perf_info.event_type
);
};
(file_name, is_retprobe, offset, cookie, ref_ctr_offset)
}
#[tag(root)]
#[test]
fn test_perf_event_link_info_uprobe_uretprobe() {
let mut obj = get_test_object("uprobe.bpf.o");
let prog: libbpf_rs::ProgramMut = get_prog_mut(&mut obj, "handle__uprobe");
let path = current_exe().expect("failed to find executable name");
let path_cstr = CString::new(path.to_str().unwrap()).ok();
let func_name = "uprobe_target";
let func_offset = get_symbol_offset(&path, func_name).unwrap();
let uprobe_opts = UprobeOpts {
ref_ctr_offset: 0,
cookie: 5,
func_name: Some(func_name.into()),
retprobe: false,
..Default::default()
};
let (up_file, up_is_retprobe, up_offset, up_cookie, up_ref_ctr_offset) =
attach_uprobe_get_info(&prog, &path, 0, &uprobe_opts);
assert_eq!(up_file, path_cstr);
assert_eq!(
up_is_retprobe, uprobe_opts.retprobe,
"Expected uprobe (not retprobe)"
);
assert_eq!(up_offset, func_offset as u32);
assert_eq!(up_cookie, uprobe_opts.cookie);
assert_eq!(up_ref_ctr_offset, uprobe_opts.ref_ctr_offset as u64);
let uretprobe_opts = UprobeOpts {
ref_ctr_offset: 0,
cookie: 13,
func_name: None,
retprobe: true,
..Default::default()
};
let (uretp_file, uretp_is_retprobe, uretp_offset, uretp_cookie, uretp_ref_ctr_offset) =
attach_uprobe_get_info(&prog, &path, func_offset, &uretprobe_opts);
assert_eq!(uretp_file, path_cstr);
assert_eq!(
uretp_is_retprobe, uretprobe_opts.retprobe,
"Expected uretprobe (not uprobe)"
);
assert_eq!(uretp_offset, func_offset as u32);
assert_eq!(uretp_cookie, uretprobe_opts.cookie);
assert_eq!(uretp_ref_ctr_offset, uretprobe_opts.ref_ctr_offset as u64);
}
#[tag(root)]
#[test]
fn test_perf_event_link_info_event() {
let mut obj = get_test_object("perf_event.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__perf_event");
let mut attr = libbpf_sys::perf_event_attr {
type_: libbpf_sys::PERF_TYPE_SOFTWARE,
size: size_of::<libbpf_sys::perf_event_attr>() as u32,
config: libbpf_sys::PERF_COUNT_SW_DUMMY as u64,
..Default::default()
};
attr.set_disabled(1);
let pid = 0;
let cpu = -1;
let group_fd = -1;
let flags = 0;
let pfd =
match unsafe { libc::syscall(libc::SYS_perf_event_open, &attr, pid, cpu, group_fd, flags) }
{
fd_raw @ 0.. => unsafe { OwnedFd::from_raw_fd(fd_raw as RawFd) },
_ => panic!(
"`perf_event_open` syscall failed: {:?}",
io::Error::last_os_error()
),
};
const PERF_COOKIE: u64 = 5;
let opts = PerfEventOpts {
cookie: PERF_COOKIE,
..Default::default()
};
let mut link = prog
.attach_perf_event_with_opts(pfd.as_raw_fd(), opts)
.expect("failed to attach perf_event");
let link_info = link.info().expect("failed to get perf_event link info");
link.disconnect();
let LinkTypeInfo::PerfEvent(perf_info) = link_info.info else {
panic!(
"Expected LinkTypeInfo::PerfEvent for perf_event, got: {:?}",
link_info.info
);
};
let PerfEventType::Event {
config,
event_type,
cookie,
} = perf_info.event_type
else {
panic!(
"Expected PerfEventType::PerfEvent, got: {:?}",
perf_info.event_type
);
};
assert_eq!(event_type, libbpf_sys::PERF_TYPE_SOFTWARE);
assert_eq!(config, libbpf_sys::PERF_COUNT_SW_DUMMY as u64);
assert_eq!(cookie, PERF_COOKIE);
}
fn buffer<'a>(perf: &'a libbpf_rs::PerfBuffer, buf_idx: usize) -> &'a [u8] {
let perf_buff_ptr = perf.as_libbpf_object();
let mut buffer_data_ptr: *mut c_void = ptr::null_mut();
let mut buffer_size: usize = 0;
let ret = unsafe {
libbpf_sys::perf_buffer__buffer(
perf_buff_ptr.as_ptr(),
buf_idx as i32,
ptr::addr_of_mut!(buffer_data_ptr),
ptr::addr_of_mut!(buffer_size) as *mut libbpf_sys::size_t,
)
};
assert!(ret >= 0);
unsafe { slice::from_raw_parts(buffer_data_ptr as *const u8, buffer_size) }
}
#[tag(root)]
#[test]
fn test_object_perf_buffer_raw() {
use memmem::Searcher;
use memmem::TwoWaySearcher;
let cookie_val = 42u16;
let mut obj = get_test_object("tracepoint.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__tracepoint_with_cookie_pb");
let opts = TracepointOpts {
cookie: cookie_val.into(),
..TracepointOpts::default()
};
let _link = prog
.attach_tracepoint_with_opts(TracepointCategory::Syscalls, "sys_enter_getpid", opts)
.expect("failed to attach prog");
let map = get_map_mut(&mut obj, "pb");
let cookie_bytes = cookie_val.to_ne_bytes();
let searcher = TwoWaySearcher::new(&cookie_bytes[..]);
let perf = libbpf_rs::PerfBufferBuilder::new(&map)
.build()
.expect("failed to build");
let _pid = unsafe { libc::getpid() };
let found_cookie = (0..perf.buffer_cnt()).any(|buf_idx| {
let buf = buffer(&perf, buf_idx);
searcher.search_in(buf).is_some()
});
assert!(found_cookie);
}
#[tag(root)]
#[test]
fn test_map_pinned_status() {
let mut obj = get_test_object("map_auto_pin.bpf.o");
let map = get_map_mut(&mut obj, "auto_pin_map");
let is_pinned = map.is_pinned();
assert!(is_pinned);
let expected_path = "/sys/fs/bpf/auto_pin_map";
let get_path = map.get_pin_path().expect("get map pin path failed");
assert_eq!(expected_path, get_path.to_str().unwrap());
let _unused = fs::remove_file(expected_path);
}
#[tag(root)]
#[test]
fn test_map_pinned_status_with_pin_root_path() {
let obj_path = get_test_object_path("map_auto_pin.bpf.o");
let mut obj = ObjectBuilder::default()
.debug(true)
.pin_root_path("/sys/fs/bpf/test_namespace")
.expect("root_pin_path failed")
.open_file(obj_path)
.expect("failed to open object")
.load()
.expect("failed to load object");
let expected_path = "/sys/fs/bpf/test_namespace/auto_pin_map";
defer! {
let _unused = fs::remove_file(expected_path);
let _unused = fs::remove_dir("/sys/fs/bpf/test_namespace");
}
let map = get_map_mut(&mut obj, "auto_pin_map");
let is_pinned = map.is_pinned();
assert!(is_pinned);
let get_path = map.get_pin_path().expect("get map pin path failed");
assert_eq!(expected_path, get_path.to_str().unwrap());
}
#[tag(root)]
#[test]
fn test_program_get_fd_and_id() {
let mut obj = get_test_object("runqslower.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__sched_wakeup");
let prog_fd = prog.as_fd();
let prog_id = Program::id_from_fd(prog_fd).expect("failed to get program id from fd");
let _owned_prog_fd = Program::fd_from_id(prog_id).expect("failed to get program fd from id");
}
#[tag(root)]
#[test]
fn test_map_autocreate_disable() {
let mut open_obj = open_test_object("map_auto_pin.bpf.o");
let mut auto_pin_map = open_obj
.maps_mut()
.find(|map| map.name() == OsStr::new("auto_pin_map"))
.expect("failed to find `auto_pin_map` map");
auto_pin_map
.set_autocreate(false)
.expect("set_autocreate() failed");
open_obj.load().expect("failed to load object");
}
#[tag(root)]
#[test]
fn test_map_autocreate_getter() {
let mut open_obj = open_test_object("map_auto_pin.bpf.o");
let map = open_obj
.maps()
.find(|m| m.name() == OsStr::new("auto_pin_map"))
.expect("failed to find `auto_pin_map` map");
assert!(map.autocreate());
let mut map = open_obj
.maps_mut()
.find(|m| m.name() == OsStr::new("auto_pin_map"))
.expect("failed to find `auto_pin_map` map");
map.set_autocreate(false).expect("set_autocreate() failed");
assert!(!map.autocreate());
let obj = open_obj.load().expect("failed to load object");
let map = obj
.maps()
.find(|m| m.name() == OsStr::new("auto_pin_map"))
.expect("failed to find `auto_pin_map` map");
assert!(!map.autocreate());
let open_obj2 = open_test_object("map_auto_pin.bpf.o");
let obj2 = open_obj2.load().expect("failed to load object");
let map = obj2
.maps()
.find(|m| m.name() == OsStr::new("auto_pin_map"))
.expect("failed to find `auto_pin_map` map");
assert!(map.autocreate());
}
#[tag(root)]
#[test]
fn test_map_query_fdinfo() {
let open_obj = open_test_object("runqslower.bpf.o");
let obj = open_obj.load().expect("failed to load object");
let map = obj
.maps()
.find(|m| m.name() == OsStr::new("start"))
.expect("failed to find `start` map");
let info = map.info().expect("info() failed");
let fdinfo = map.query_fdinfo().expect("query_fdinfo() failed");
assert_eq!(fdinfo.map_type, map.map_type());
assert_eq!(fdinfo.key_size, map.key_size());
assert_eq!(fdinfo.value_size, map.value_size());
assert_eq!(fdinfo.max_entries, map.max_entries());
assert_eq!(fdinfo.map_flags.unwrap(), info.info.map_flags);
assert_eq!(fdinfo.map_extra.unwrap(), info.info.map_extra);
assert!(fdinfo.memlock.unwrap() > 0);
assert_eq!(fdinfo.map_id.unwrap(), info.info.id);
assert!(!fdinfo.frozen.unwrap());
assert_eq!(fdinfo.owner_prog_type, None);
assert_eq!(fdinfo.owner_jited, None);
}
#[tag(root)]
#[test]
fn test_map_adjust_value_size() {
let mut open_obj = open_test_object("map_auto_pin.bpf.o");
let mut resizable = open_obj
.maps_mut()
.find(|map| map.name() == OsStr::new(".data.resizable_data"))
.expect("failed to find `.data.resizable_data` map");
let len = resizable.initial_value().unwrap().len();
assert_eq!(len, size_of::<u64>());
let () = resizable
.set_value_size(len as u32 * 2)
.expect("failed to set value size");
let new_len = resizable.initial_value().unwrap().len();
assert_eq!(new_len, len * 2);
}
#[tag(root)]
#[test]
fn test_object_map_max_entries() {
let mut obj = open_test_object("runqslower.bpf.o");
let mut start = obj
.maps_mut()
.find(|map| map.name() == OsStr::new("start"))
.expect("failed to find `start` map");
let initial_max_entries = start.max_entries();
let new_max_entries = initial_max_entries * 2;
start
.set_max_entries(new_max_entries)
.expect("failed to set max entries");
assert_eq!(start.max_entries(), new_max_entries);
let obj = obj.load().expect("failed to load object");
let start = obj
.maps()
.find(|map| map.name() == OsStr::new("start"))
.expect("failed to find `start` map");
assert_eq!(start.max_entries(), new_max_entries);
let start = MapHandle::from_map_id(start.info().expect("failed to get map info").info.id)
.expect("failed to get map handle from id");
assert!(start.max_entries() == new_max_entries);
}
#[tag(root)]
#[test]
fn test_attach_ksyscall() {
let mut obj = get_test_object("ksyscall.bpf.o");
let prog = get_prog_mut(&mut obj, "handle__ksyscall");
let _link = prog
.attach_ksyscall(false, "kill")
.expect("failed to attach prog");
let map = get_map_mut(&mut obj, "ringbuf");
let action = || {
let ret = unsafe { libc::kill(libc::getpid(), libc::SIGCHLD) };
if ret < 0 {
panic!("kill failed: {}", io::Error::last_os_error());
}
};
let result = with_ringbuffer(&map, action);
assert_eq!(result, 1);
}
#[tag(root)]
#[test]
fn test_run_prog_success() {
let mut obj = get_test_object("run_prog.bpf.o");
let prog = get_prog_mut(&mut obj, "test_1");
#[repr(C)]
struct bpf_dummy_ops_state {
val: c_int,
}
let value = 42;
let state = bpf_dummy_ops_state { val: value };
let mut args = [addr_of!(state) as u64];
let input = ProgramInput {
context_in: Some(unsafe {
slice::from_raw_parts_mut(&mut args as *mut _ as *mut u8, size_of_val(&args))
}),
..Default::default()
};
let output = prog.test_run(input).unwrap();
assert_eq!(output.return_value, value as _);
}
#[tag(root)]
#[test]
fn test_run_prog_fail() {
let mut obj = get_test_object("run_prog.bpf.o");
let prog = get_prog_mut(&mut obj, "test_2");
let input = ProgramInput::default();
let _err = prog.test_run(input).unwrap_err();
}
#[tag(root)]
#[test]
fn test_run_prog_repeat_and_duration() {
let repeat = 100;
let payload: [u8; 16] = [
0xAA, 0xBB, 0xCC, 0xDD, 0xEE, 0xFF, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x08, 0x00, 0x00, 0x00, ];
let mut obj = get_test_object("run_prog.bpf.o");
let prog = get_prog_mut(&mut obj, "xdp_counter");
let input: ProgramInput<'_> = ProgramInput {
data_in: Some(&payload),
repeat,
..Default::default()
};
let output = prog.test_run(input).unwrap();
let map = get_map(&obj, "test_counter_map");
let counter = map
.lookup(&0u32.to_ne_bytes(), MapFlags::ANY)
.expect("failed to lookup counter")
.expect("failed to retrieve value");
assert_eq!(output.return_value, libbpf_sys::XDP_PASS);
assert_eq!(
counter,
repeat.to_ne_bytes(),
"counter {} != repeat {repeat}",
u32::from_ne_bytes(counter.clone().try_into().unwrap())
);
assert_ne!(
output.duration,
Duration::ZERO,
"duration should be non-zero"
);
}
#[tag(root)]
#[test]
#[ignore = "requires kernel with struct_ops association support"]
fn test_assoc_struct_ops() {
let obj_for_map = get_test_object("run_prog.bpf.o");
let map = get_map(&obj_for_map, "dummy_1");
let mut obj_for_prog = get_test_object("run_prog.bpf.o");
let prog = get_prog_mut(&mut obj_for_prog, "xdp_counter");
prog.assoc_struct_ops(&map).unwrap();
}
#[tag(root)]
#[test]
fn test_prog_info_verified_insns() {
let obj = get_test_object("tracepoint.bpf.o");
let prog = obj
.progs()
.next()
.expect("should have at least one program");
let prog_id = Program::id_from_fd(prog.as_fd()).expect("failed to get program ID");
let info = ProgInfoIter::default()
.find(|p| p.id == prog_id)
.unwrap_or_else(|| panic!("failed to find program with ID {prog_id} via ProgInfoIter"));
assert!(
info.verified_insns > 0,
"expected verified_insns > 0, got {}",
info.verified_insns,
);
}