Add integration test for perf link pin

Add integration testing for link pinning and
loading/unloading of tracepoint, kprobe, and
uprobe programs.

Redo how we utilize bpftool to verify that programs
are loaded to be explicit with names. Also add a helper
to verify that a program is loaded AND linked.

Signed-off-by: Andrew Stoycos <astoycos@redhat.com>
pull/560/head
Andrew Stoycos 2 years ago
parent e668ae0c1b
commit fb075636c1
No known key found for this signature in database
GPG Key ID: 66735B92BB71C096

@ -3,11 +3,11 @@
use aya_bpf::{ use aya_bpf::{
bindings::xdp_action, bindings::xdp_action,
macros::{kprobe, xdp}, macros::{kprobe, tracepoint, uprobe, xdp},
programs::{ProbeContext, XdpContext}, programs::{ProbeContext, TracePointContext, XdpContext},
}; };
#[xdp(name = "test_unload_xdp")] #[xdp(name = "test_xdp")]
pub fn pass(ctx: XdpContext) -> u32 { pub fn pass(ctx: XdpContext) -> u32 {
match unsafe { try_pass(ctx) } { match unsafe { try_pass(ctx) } {
Ok(ret) => ret, Ok(ret) => ret,
@ -20,8 +20,17 @@ unsafe fn try_pass(_ctx: XdpContext) -> Result<u32, u32> {
} }
#[kprobe] #[kprobe]
// truncated name to match bpftool output pub fn test_kprobe(_ctx: ProbeContext) -> u32 {
pub fn test_unload_kpr(_ctx: ProbeContext) -> u32 { 0
}
#[tracepoint]
pub fn test_tracepoint(_ctx: TracePointContext) -> u32 {
0
}
#[uprobe]
pub fn test_uprobe(_ctx: ProbeContext) -> u32 {
0 0
} }

@ -1,10 +1,10 @@
use std::{convert::TryInto as _, thread, time}; use std::{convert::TryInto as _, process::Command, thread, time};
use aya::{ use aya::{
maps::Array, maps::Array,
programs::{ programs::{
links::{FdLink, PinnedLink}, links::{FdLink, PinnedLink},
loaded_programs, KProbe, TracePoint, Xdp, XdpFlags, loaded_programs, KProbe, TracePoint, UProbe, Xdp, XdpFlags,
}, },
util::KernelVersion, util::KernelVersion,
Bpf, Bpf,
@ -50,6 +50,40 @@ fn multiple_btf_maps() {
assert_eq!(val_2, 42); assert_eq!(val_2, 42);
} }
fn is_linked(prog_id: &u32) -> bool {
let output = Command::new("bpftool").args(["link"]).output();
let output = output.expect("Failed to run 'bpftool link'");
let stdout = String::from_utf8(output.stdout).unwrap();
stdout.contains(&prog_id.to_string())
}
macro_rules! assert_loaded_and_linked {
($name:literal, $loaded:expr) => {
for i in 0..(MAX_RETRIES + 1) {
let id = loaded_programs()
.find(|prog| prog.as_ref().unwrap().name() == $name.as_bytes())
.map(|prog| Some(prog.unwrap().id()));
let mut linked = false;
if let Some(prog_id) = id {
linked = is_linked(&prog_id.unwrap());
if linked == $loaded {
break;
}
}
if i == MAX_RETRIES {
panic!(
"Expected (loaded/linked: {}) but found (id: {}, linked: {}",
$loaded,
id.is_some(),
linked
);
}
thread::sleep(time::Duration::from_millis(RETRY_DURATION_MS));
}
};
}
macro_rules! assert_loaded { macro_rules! assert_loaded {
($name:literal, $loaded:expr) => { ($name:literal, $loaded:expr) => {
for i in 0..(MAX_RETRIES + 1) { for i in 0..(MAX_RETRIES + 1) {
@ -68,59 +102,109 @@ macro_rules! assert_loaded {
#[test] #[test]
fn unload_xdp() { fn unload_xdp() {
let mut bpf = Bpf::load(crate::TEST).unwrap(); let mut bpf = Bpf::load(crate::TEST).unwrap();
let prog: &mut Xdp = bpf let prog: &mut Xdp = bpf.program_mut("test_xdp").unwrap().try_into().unwrap();
.program_mut("test_unload_xdp")
.unwrap()
.try_into()
.unwrap();
prog.load().unwrap(); prog.load().unwrap();
assert_loaded!("test_unload_xdp", true); assert_loaded!("test_xdp", true);
let link = prog.attach("lo", XdpFlags::default()).unwrap(); let link = prog.attach("lo", XdpFlags::default()).unwrap();
{ {
let _link_owned = prog.take_link(link).unwrap(); let _link_owned = prog.take_link(link).unwrap();
prog.unload().unwrap(); prog.unload().unwrap();
assert_loaded!("test_unload_xdp", true); assert_loaded_and_linked!("test_xdp", true);
}; };
assert_loaded!("test_unload_xdp", false); assert_loaded!("test_xdp", false);
prog.load().unwrap(); prog.load().unwrap();
assert_loaded!("test_unload_xdp", true); assert_loaded!("test_xdp", true);
prog.attach("lo", XdpFlags::default()).unwrap(); prog.attach("lo", XdpFlags::default()).unwrap();
assert_loaded!("test_unload_xdp", true); assert_loaded!("test_xdp", true);
prog.unload().unwrap(); prog.unload().unwrap();
assert_loaded!("test_unload_xdp", false); assert_loaded!("test_xdp", false);
} }
#[test] #[test]
fn unload_kprobe() { fn unload_kprobe() {
let mut bpf = Bpf::load(crate::TEST).unwrap(); let mut bpf = Bpf::load(crate::TEST).unwrap();
let prog: &mut KProbe = bpf let prog: &mut KProbe = bpf.program_mut("test_kprobe").unwrap().try_into().unwrap();
.program_mut("test_unload_kpr") prog.load().unwrap();
assert_loaded!("test_kprobe", true);
let link = prog.attach("try_to_wake_up", 0).unwrap();
{
let _link_owned = prog.take_link(link).unwrap();
prog.unload().unwrap();
assert_loaded_and_linked!("test_kprobe", true);
};
assert_loaded!("test_kprobe", false);
prog.load().unwrap();
assert_loaded!("test_kprobe", true);
prog.attach("try_to_wake_up", 0).unwrap();
assert_loaded!("test_kprobe", true);
prog.unload().unwrap();
assert_loaded!("test_kprobe", false);
}
#[test]
fn basic_tracepoint() {
let mut bpf = Bpf::load(crate::TEST).unwrap();
let prog: &mut TracePoint = bpf
.program_mut("test_tracepoint")
.unwrap() .unwrap()
.try_into() .try_into()
.unwrap(); .unwrap();
prog.load().unwrap(); prog.load().unwrap();
assert_loaded!("test_unload_kpr", true); assert_loaded!("test_tracepoint", true);
let link = prog.attach("try_to_wake_up", 0).unwrap(); let link = prog.attach("syscalls", "sys_enter_kill").unwrap();
{ {
let _link_owned = prog.take_link(link).unwrap(); let _link_owned = prog.take_link(link).unwrap();
prog.unload().unwrap(); prog.unload().unwrap();
assert_loaded!("test_unload_kpr", true); assert_loaded_and_linked!("test_tracepoint", true);
}; };
assert_loaded!("test_unload_kpr", false); assert_loaded!("test_tracepoint", false);
prog.load().unwrap(); prog.load().unwrap();
assert_loaded!("test_unload_kpr", true); assert_loaded!("test_tracepoint", true);
prog.attach("try_to_wake_up", 0).unwrap(); prog.attach("syscalls", "sys_enter_kill").unwrap();
assert_loaded!("test_tracepoint", true);
prog.unload().unwrap();
assert_loaded!("test_tracepoint", false);
}
#[test]
fn basic_uprobe() {
let mut bpf = Bpf::load(crate::TEST).unwrap();
let prog: &mut UProbe = bpf.program_mut("test_uprobe").unwrap().try_into().unwrap();
prog.load().unwrap();
assert_loaded!("test_uprobe", true);
let link = prog.attach(Some("sleep"), 0, "libc", None).unwrap();
{
let _link_owned = prog.take_link(link).unwrap();
prog.unload().unwrap();
assert_loaded_and_linked!("test_uprobe", true);
};
assert_loaded!("test_uprobe", false);
prog.load().unwrap();
assert_loaded!("test_unload_kpr", true); assert_loaded!("test_uprobe", true);
prog.attach(Some("sleep"), 0, "libc", None).unwrap();
assert_loaded!("test_uprobe", true);
prog.unload().unwrap(); prog.unload().unwrap();
assert_loaded!("test_unload_kpr", false); assert_loaded!("test_uprobe", false);
} }
#[test] #[test]
@ -132,30 +216,26 @@ fn pin_link() {
} }
let mut bpf = Bpf::load(crate::TEST).unwrap(); let mut bpf = Bpf::load(crate::TEST).unwrap();
let prog: &mut Xdp = bpf let prog: &mut Xdp = bpf.program_mut("test_xdp").unwrap().try_into().unwrap();
.program_mut("test_unload_xdp")
.unwrap()
.try_into()
.unwrap();
prog.load().unwrap(); prog.load().unwrap();
let link_id = prog.attach("lo", XdpFlags::default()).unwrap(); let link_id = prog.attach("lo", XdpFlags::default()).unwrap();
let link = prog.take_link(link_id).unwrap(); let link = prog.take_link(link_id).unwrap();
assert_loaded!("test_unload_xdp", true); assert_loaded!("test_xdp", true);
let fd_link: FdLink = link.try_into().unwrap(); let fd_link: FdLink = link.try_into().unwrap();
let pinned = fd_link.pin("/sys/fs/bpf/aya-xdp-test-lo").unwrap(); let pinned = fd_link.pin("/sys/fs/bpf/aya-xdp-test-lo").unwrap();
// because of the pin, the program is still attached // because of the pin, the program is still attached
prog.unload().unwrap(); prog.unload().unwrap();
assert_loaded!("test_unload_xdp", true); assert_loaded!("test_xdp", true);
// delete the pin, but the program is still attached // delete the pin, but the program is still attached
let new_link = pinned.unpin().unwrap(); let new_link = pinned.unpin().unwrap();
assert_loaded!("test_unload_xdp", true); assert_loaded!("test_xdp", true);
// finally when new_link is dropped we're detached // finally when new_link is dropped we're detached
drop(new_link); drop(new_link);
assert_loaded!("test_unload_xdp", false); assert_loaded!("test_xdp", false);
} }
#[test] #[test]
@ -198,7 +278,7 @@ fn pin_lifecycle() {
} }
// should still be loaded since link was pinned // should still be loaded since link was pinned
assert_loaded!("pass", true); assert_loaded_and_linked!("pass", true);
// 4. Load a new version of the program, unpin link, and atomically replace old program // 4. Load a new version of the program, unpin link, and atomically replace old program
{ {
@ -217,3 +297,173 @@ fn pin_lifecycle() {
// program should be unloaded // program should be unloaded
assert_loaded!("pass", false); assert_loaded!("pass", false);
} }
#[test]
fn pin_lifecycle_tracepoint() {
// 1. Load Program and Pin
{
let mut bpf = Bpf::load(crate::TEST).unwrap();
let prog: &mut TracePoint = bpf
.program_mut("test_tracepoint")
.unwrap()
.try_into()
.unwrap();
prog.load().unwrap();
prog.pin("/sys/fs/bpf/aya-tracepoint-test-prog").unwrap();
}
// should still be loaded since prog was pinned
assert_loaded!("test_tracepoint", true);
// 2. Load program from bpffs but don't attach it
{
let _ = TracePoint::from_pin("/sys/fs/bpf/aya-tracepoint-test-prog").unwrap();
}
// should still be loaded since prog was pinned
assert_loaded!("test_tracepoint", true);
// 3. Load program from bpffs and attach
{
let mut prog = TracePoint::from_pin("/sys/fs/bpf/aya-tracepoint-test-prog").unwrap();
let link_id = prog.attach("syscalls", "sys_enter_kill").unwrap();
let link = prog.take_link(link_id).unwrap();
let fd_link: FdLink = link.try_into().unwrap();
fd_link
.pin("/sys/fs/bpf/aya-tracepoint-test-sys-enter-kill")
.unwrap();
// Unpin the program. It will stay attached since its links were pinned.
prog.unpin().unwrap();
}
// should still be loaded since link was pinned
assert_loaded_and_linked!("test_tracepoint", true);
// 4. unpin link, and make sure everything is unloaded
{
PinnedLink::from_pin("/sys/fs/bpf/aya-tracepoint-test-sys-enter-kill")
.unwrap()
.unpin()
.unwrap();
}
// program should be unloaded
assert_loaded!("test_tracepoint", false);
}
#[test]
fn pin_lifecycle_kprobe() {
// 1. Load Program and Pin
{
let mut bpf = Bpf::load(crate::TEST).unwrap();
let prog: &mut KProbe = bpf.program_mut("test_kprobe").unwrap().try_into().unwrap();
prog.load().unwrap();
prog.pin("/sys/fs/bpf/aya-kprobe-test-prog").unwrap();
}
// should still be loaded since prog was pinned
assert_loaded!("test_kprobe", true);
// 2. Load program from bpffs but don't attach it
{
let _ = KProbe::from_pin(
"/sys/fs/bpf/aya-kprobe-test-prog",
aya::programs::ProbeKind::KProbe,
)
.unwrap();
}
// should still be loaded since prog was pinned
assert_loaded!("test_kprobe", true);
// 3. Load program from bpffs and attach
{
let mut prog = KProbe::from_pin(
"/sys/fs/bpf/aya-kprobe-test-prog",
aya::programs::ProbeKind::KProbe,
)
.unwrap();
let link_id = prog.attach("try_to_wake_up", 0).unwrap();
let link = prog.take_link(link_id).unwrap();
let fd_link: FdLink = link.try_into().unwrap();
fd_link
.pin("/sys/fs/bpf/aya-kprobe-test-try-to-wake-up")
.unwrap();
// Unpin the program. It will stay attached since its links were pinned.
prog.unpin().unwrap();
}
// should still be loaded since link was pinned
assert_loaded_and_linked!("test_kprobe", true);
// 4. unpin link, and make sure everything is unloaded
{
PinnedLink::from_pin("/sys/fs/bpf/aya-kprobe-test-try-to-wake-up")
.unwrap()
.unpin()
.unwrap();
}
// program should be unloaded
assert_loaded!("test_kprobe", false);
}
#[test]
fn pin_lifecycle_uprobe() {
// 1. Load Program and Pin
{
let mut bpf = Bpf::load(crate::TEST).unwrap();
let prog: &mut UProbe = bpf.program_mut("test_uprobe").unwrap().try_into().unwrap();
prog.load().unwrap();
prog.pin("/sys/fs/bpf/aya-uprobe-test-prog").unwrap();
}
// should still be loaded since prog was pinned
assert_loaded!("test_uprobe", true);
// 2. Load program from bpffs but don't attach it
{
let _ = UProbe::from_pin(
"/sys/fs/bpf/aya-uprobe-test-prog",
aya::programs::ProbeKind::UProbe,
)
.unwrap();
}
// should still be loaded since prog was pinned
assert_loaded!("test_uprobe", true);
// 3. Load program from bpffs and attach
{
let mut prog = UProbe::from_pin(
"/sys/fs/bpf/aya-uprobe-test-prog",
aya::programs::ProbeKind::UProbe,
)
.unwrap();
let link_id = prog.attach(Some("sleep"), 0, "libc", None).unwrap();
let link = prog.take_link(link_id).unwrap();
let fd_link: FdLink = link.try_into().unwrap();
fd_link
.pin("/sys/fs/bpf/aya-uprobe-test-bash-sleep")
.unwrap();
// Unpin the program. It will stay attached since its links were pinned.
prog.unpin().unwrap();
}
// should still be loaded since link was pinned
assert_loaded_and_linked!("test_uprobe", true);
// 4. unpin link, and make sure everything is unloaded
{
PinnedLink::from_pin("/sys/fs/bpf/aya-uprobe-test-bash-sleep")
.unwrap()
.unpin()
.unwrap();
}
// program should be unloaded
assert_loaded!("test_uprobe", false);
}

Loading…
Cancel
Save