aya: support both bpf_map_def layout variants

Libbpf and iproute2 use two slightly different `bpf_map_def` layouts. This change implements support for loading both.

Refs: #10, #14
pull/15/head
Rafael Ortiz 4 years ago committed by GitHub
parent 97cb0db0ad
commit d8d311738c
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

@ -48,13 +48,15 @@ unsafe_impl_pod!(i8, u8, i16, u16, i32, u32, i64, u64);
#[allow(non_camel_case_types)] #[allow(non_camel_case_types)]
#[repr(C)] #[repr(C)]
#[derive(Copy, Clone, Debug)] #[derive(Copy, Clone, Debug, Default, PartialEq)]
pub(crate) struct bpf_map_def { pub(crate) struct bpf_map_def {
// minimum features required by old BPF programs
pub(crate) map_type: u32, pub(crate) map_type: u32,
pub(crate) key_size: u32, pub(crate) key_size: u32,
pub(crate) value_size: u32, pub(crate) value_size: u32,
pub(crate) max_entries: u32, pub(crate) max_entries: u32,
pub(crate) map_flags: u32, pub(crate) map_flags: u32,
// optional features
pub(crate) id: u32, pub(crate) id: u32,
pub(crate) pinning: u32, pub(crate) pinning: u32,
} }

@ -162,9 +162,7 @@ mod tests {
key_size: 4, key_size: 4,
value_size: 4, value_size: 4,
max_entries: 1024, max_entries: 1024,
map_flags: 0, ..Default::default()
id: 0,
pinning: 0,
}, },
section_index: 0, section_index: 0,
data: Vec::new(), data: Vec::new(),
@ -215,9 +213,7 @@ mod tests {
key_size: 4, key_size: 4,
value_size: 4, value_size: 4,
max_entries: 1024, max_entries: 1024,
map_flags: 0, ..Default::default()
id: 0,
pinning: 0,
}, },
section_index: 0, section_index: 0,
data: Vec::new(), data: Vec::new(),
@ -273,9 +269,7 @@ mod tests {
key_size: 4, key_size: 4,
value_size: 4, value_size: 4,
max_entries: 1024, max_entries: 1024,
map_flags: 0, ..Default::default()
id: 0,
pinning: 0,
}, },
section_index: 0, section_index: 0,
data: Vec::new(), data: Vec::new(),

@ -426,9 +426,7 @@ mod tests {
key_size: 4, key_size: 4,
value_size: 4, value_size: 4,
max_entries: 1024, max_entries: 1024,
map_flags: 0, ..Default::default()
id: 0,
pinning: 0,
}, },
section_index: 0, section_index: 0,
data: Vec::new(), data: Vec::new(),

@ -22,8 +22,11 @@ use crate::{
obj::btf::{Btf, BtfError, BtfExt}, obj::btf::{Btf, BtfError, BtfExt},
BpfError, BpfError,
}; };
use std::slice::from_raw_parts_mut;
const KERNEL_VERSION_ANY: u32 = 0xFFFF_FFFE; const KERNEL_VERSION_ANY: u32 = 0xFFFF_FFFE;
/// The first five __u32 of `bpf_map_def` must be defined.
const MINIMUM_MAP_SIZE: usize = mem::size_of::<u32>() * 5;
#[derive(Clone)] #[derive(Clone)]
pub struct Object { pub struct Object {
@ -493,8 +496,7 @@ fn parse_map(section: &Section, name: &str) -> Result<Map, ParseError> {
value_size: section.data.len() as u32, value_size: section.data.len() as u32,
max_entries: 1, max_entries: 1,
map_flags: 0, /* FIXME: set rodata readonly */ map_flags: 0, /* FIXME: set rodata readonly */
id: 0, ..Default::default()
pinning: 0,
}; };
(def, section.data.to_vec()) (def, section.data.to_vec())
} else { } else {
@ -510,13 +512,23 @@ fn parse_map(section: &Section, name: &str) -> Result<Map, ParseError> {
} }
fn parse_map_def(name: &str, data: &[u8]) -> Result<bpf_map_def, ParseError> { fn parse_map_def(name: &str, data: &[u8]) -> Result<bpf_map_def, ParseError> {
if mem::size_of::<bpf_map_def>() > data.len() { if data.len() > mem::size_of::<bpf_map_def>() || data.len() < MINIMUM_MAP_SIZE {
return Err(ParseError::InvalidMapDefinition { return Err(ParseError::InvalidMapDefinition {
name: name.to_owned(), name: name.to_owned(),
}); });
} }
Ok(unsafe { ptr::read_unaligned(data.as_ptr() as *const bpf_map_def) }) if data.len() < mem::size_of::<bpf_map_def>() {
let mut map_def = bpf_map_def::default();
unsafe {
let map_def_ptr =
from_raw_parts_mut(&mut map_def as *mut bpf_map_def as *mut u8, data.len());
map_def_ptr.copy_from_slice(data);
}
Ok(map_def)
} else {
Ok(unsafe { ptr::read_unaligned(data.as_ptr() as *const bpf_map_def) })
}
} }
fn copy_instructions(data: &[u8]) -> Result<Vec<bpf_insn>, ParseError> { fn copy_instructions(data: &[u8]) -> Result<Vec<bpf_insn>, ParseError> {
@ -662,6 +674,10 @@ mod tests {
Err(ParseError::InvalidMapDefinition { .. }) Err(ParseError::InvalidMapDefinition { .. })
)); ));
assert!(matches!( assert!(matches!(
parse_map_def("foo", &[0u8; std::mem::size_of::<bpf_map_def>() + 1]),
Err(ParseError::InvalidMapDefinition { .. })
));
assert_eq!(
parse_map_def( parse_map_def(
"foo", "foo",
bytes_of(&bpf_map_def { bytes_of(&bpf_map_def {
@ -670,20 +686,55 @@ mod tests {
value_size: 3, value_size: 3,
max_entries: 4, max_entries: 4,
map_flags: 5, map_flags: 5,
id: 0, ..Default::default()
pinning: 0
}) })
), )
Ok(bpf_map_def { .unwrap(),
bpf_map_def {
map_type: 1, map_type: 1,
key_size: 2, key_size: 2,
value_size: 3, value_size: 3,
max_entries: 4, max_entries: 4,
map_flags: 5, map_flags: 5,
id: 0, ..Default::default()
pinning: 0 }
}) );
));
assert_eq!(
parse_map_def(
"foo",
&bytes_of(&bpf_map_def {
map_type: 1,
key_size: 2,
value_size: 3,
max_entries: 4,
map_flags: 5,
..Default::default()
})[..(mem::size_of::<u32>() * 5)]
)
.unwrap(),
bpf_map_def {
map_type: 1,
key_size: 2,
value_size: 3,
max_entries: 4,
map_flags: 5,
..Default::default()
}
);
let map = parse_map_def(
"foo",
&bytes_of(&bpf_map_def {
map_type: 1,
key_size: 2,
value_size: 3,
max_entries: 4,
map_flags: 5,
..Default::default()
})[..(mem::size_of::<u32>() * 5)],
)
.unwrap();
assert!(map.id == 0 && map.pinning == 0)
} }
#[test] #[test]
@ -691,6 +742,13 @@ mod tests {
assert!(matches!( assert!(matches!(
parse_map(&fake_section("maps/foo", &[]), "foo"), parse_map(&fake_section("maps/foo", &[]), "foo"),
Err(ParseError::InvalidMapDefinition { .. }) Err(ParseError::InvalidMapDefinition { .. })
));
assert!(matches!(
parse_map(
&fake_section("maps/foo", &[0u8; std::mem::size_of::<bpf_map_def>() + 1]),
"foo"
),
Err(ParseError::InvalidMapDefinition { .. })
)) ))
} }
@ -750,7 +808,7 @@ mod tests {
max_entries: 1, max_entries: 1,
map_flags: 0, map_flags: 0,
id: 0, id: 0,
pinning: 0 pinning: 0,
}, },
data data
}) if name == ".bss" && data == map_data && value_size == map_data.len() as u32 }) if name == ".bss" && data == map_data && value_size == map_data.len() as u32
@ -809,8 +867,7 @@ mod tests {
value_size: 3, value_size: 3,
max_entries: 4, max_entries: 4,
map_flags: 5, map_flags: 5,
id: 0, ..Default::default()
pinning: 0
}) })
),), ),),
Ok(()) Ok(())

Loading…
Cancel
Save