aya: allow global value to be optional

This allow to not error out when a global symbol is missing from the object.
pull/632/head
Mary 2 years ago
parent 4c08b9b43f
commit 93435fc854

@ -597,7 +597,10 @@ impl Object {
} }
/// Patches map data /// Patches map data
pub fn patch_map_data(&mut self, globals: HashMap<&str, &[u8]>) -> Result<(), ParseError> { pub fn patch_map_data(
&mut self,
globals: HashMap<&str, (&[u8], bool)>,
) -> Result<(), ParseError> {
let symbols: HashMap<String, &Symbol> = self let symbols: HashMap<String, &Symbol> = self
.symbol_table .symbol_table
.iter() .iter()
@ -605,7 +608,7 @@ impl Object {
.map(|(_, s)| (s.name.as_ref().unwrap().clone(), s)) .map(|(_, s)| (s.name.as_ref().unwrap().clone(), s))
.collect(); .collect();
for (name, data) in globals { for (name, (data, must_exist)) in globals {
if let Some(symbol) = symbols.get(name) { if let Some(symbol) = symbols.get(name) {
if data.len() as u64 != symbol.size { if data.len() as u64 != symbol.size {
return Err(ParseError::InvalidGlobalData { return Err(ParseError::InvalidGlobalData {
@ -633,7 +636,7 @@ impl Object {
}); });
} }
map.data_mut().splice(start..end, data.iter().cloned()); map.data_mut().splice(start..end, data.iter().cloned());
} else { } else if must_exist {
return Err(ParseError::SymbolNotFound { return Err(ParseError::SymbolNotFound {
name: name.to_owned(), name: name.to_owned(),
}); });
@ -2354,8 +2357,11 @@ mod tests {
); );
let test_data: &[u8] = &[1, 2, 3]; let test_data: &[u8] = &[1, 2, 3];
obj.patch_map_data(HashMap::from([("my_config", test_data)])) obj.patch_map_data(HashMap::from([
.unwrap(); ("my_config", (test_data, true)),
("optional_variable", (test_data, false)),
]))
.unwrap();
let map = obj.maps.get(".rodata").unwrap(); let map = obj.maps.get(".rodata").unwrap();
assert_eq!(test_data, map.data()); assert_eq!(test_data, map.data());

@ -119,7 +119,7 @@ fn detect_features() -> Features {
pub struct BpfLoader<'a> { pub struct BpfLoader<'a> {
btf: Option<Cow<'a, Btf>>, btf: Option<Cow<'a, Btf>>,
map_pin_path: Option<PathBuf>, map_pin_path: Option<PathBuf>,
globals: HashMap<&'a str, &'a [u8]>, globals: HashMap<&'a str, (&'a [u8], bool)>,
max_entries: HashMap<&'a str, u32>, max_entries: HashMap<&'a str, u32>,
extensions: HashSet<&'a str>, extensions: HashSet<&'a str>,
verifier_log_level: VerifierLogLevel, verifier_log_level: VerifierLogLevel,
@ -203,6 +203,7 @@ impl<'a> BpfLoader<'a> {
} }
/// Sets the value of a global variable. /// Sets the value of a global variable.
/// If the `must_exist` argument is `true`, [`BpfLoader::load`] will fail with [`ParseError::SymbolNotFound`] if the loaded object code does not contain the variable.
/// ///
/// From Rust eBPF, a global variable can be defined as follows: /// From Rust eBPF, a global variable can be defined as follows:
/// ///
@ -233,8 +234,8 @@ impl<'a> BpfLoader<'a> {
/// use aya::BpfLoader; /// use aya::BpfLoader;
/// ///
/// let bpf = BpfLoader::new() /// let bpf = BpfLoader::new()
/// .set_global("VERSION", &2) /// .set_global("VERSION", &2, true)
/// .set_global("PIDS", &[1234u16, 5678]) /// .set_global("PIDS", &[1234u16, 5678], true)
/// .load_file("file.o")?; /// .load_file("file.o")?;
/// # Ok::<(), aya::BpfError>(()) /// # Ok::<(), aya::BpfError>(())
/// ``` /// ```
@ -243,8 +244,9 @@ impl<'a> BpfLoader<'a> {
&mut self, &mut self,
name: &'a str, name: &'a str,
value: T, value: T,
must_exist: bool,
) -> &mut BpfLoader<'a> { ) -> &mut BpfLoader<'a> {
self.globals.insert(name, value.into().bytes); self.globals.insert(name, (value.into().bytes, must_exist));
self self
} }

Loading…
Cancel
Save