Skip to content

Commit

Permalink
Generate mutable and shared datasec accessors
Browse files Browse the repository at this point in the history
It's not particularly user friendly to only have exclusive shared
datasec accessor variants available, as that can lead to unnecessary
borrow conflicts.
With this change we revamp the generation logic to generate shared and
exclusive accessors when possible.

Refs: #606

Signed-off-by: Daniel Müller <[email protected]>
  • Loading branch information
d-e-s-o committed Nov 20, 2023
1 parent 4005445 commit 16e87b1
Show file tree
Hide file tree
Showing 6 changed files with 55 additions and 28 deletions.
6 changes: 3 additions & 3 deletions examples/capable/src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -186,9 +186,9 @@ fn main() -> Result<()> {

let mut open_skel = skel_builder.open()?;
//Pass configuration to BPF
open_skel.rodata().tool_config.tgid = opts.pid; //tgid in kernel is pid in userland
open_skel.rodata().tool_config.verbose = opts.verbose;
open_skel.rodata().tool_config.unique_type = opts.unique_type;
open_skel.rodata_mut().tool_config.tgid = opts.pid; //tgid in kernel is pid in userland
open_skel.rodata_mut().tool_config.verbose = opts.verbose;
open_skel.rodata_mut().tool_config.unique_type = opts.unique_type;

let mut skel = open_skel.load()?;
skel.attach()?;
Expand Down
6 changes: 3 additions & 3 deletions examples/runqslower/src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -90,9 +90,9 @@ fn main() -> Result<()> {
let mut open_skel = skel_builder.open()?;

// Write arguments into prog
open_skel.rodata().min_us = opts.latency;
open_skel.rodata().targ_pid = opts.pid;
open_skel.rodata().targ_tgid = opts.tid;
open_skel.rodata_mut().min_us = opts.latency;
open_skel.rodata_mut().targ_pid = opts.pid;
open_skel.rodata_mut().targ_tgid = opts.tid;

// Begin tracing
let mut skel = open_skel.load()?;
Expand Down
6 changes: 3 additions & 3 deletions examples/tproxy/src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -64,9 +64,9 @@ fn main() -> Result<()> {

// Set constants
let mut open_skel = skel_builder.open()?;
open_skel.rodata().target_port = opts.port.to_be();
open_skel.rodata().proxy_addr = proxy_addr.to_be();
open_skel.rodata().proxy_port = opts.proxy_port.to_be();
open_skel.rodata_mut().target_port = opts.port.to_be();
open_skel.rodata_mut().proxy_addr = proxy_addr.to_be();
open_skel.rodata_mut().proxy_port = opts.proxy_port.to_be();

// Load into kernel
let skel = open_skel.load()?;
Expand Down
6 changes: 6 additions & 0 deletions libbpf-cargo/CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,3 +1,9 @@
Unreleased
----------
- Adjusted skeleton creation logic to generate shared and exclusive datasec
accessor functions


0.21.2
------
- Added `Default` impl for generated `struct` types containing pointers
Expand Down
37 changes: 23 additions & 14 deletions libbpf-cargo/src/gen/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -530,24 +530,33 @@ fn gen_skel_datasec_getters(
None => continue,
};
let struct_name = format!("{obj_name}_{name}_types::{name}");
let mutability = if loaded && map_is_readonly(map) {
""
let immutable = loaded && map_is_readonly(map);
let mutabilities = if immutable {
[false].as_ref()
} else {
"mut"
[false, true].as_ref()
};

write!(
skel,
r#"
pub fn {name}(&{mutability} self) -> &{mutability} {struct_name} {{
unsafe {{
std::mem::transmute::<*mut std::ffi::c_void, &{mutability} {struct_name}>(
self.skel_config.map_mmap_ptr({idx}).unwrap()
)
for mutable in mutabilities {
let (ref_suffix, ptr_suffix, fn_suffix) = if *mutable {
("mut", "mut", "_mut")
} else {
("", "const", "")
};

write!(
skel,
r#"
pub fn {name}{fn_suffix}(&{ref_suffix} self) -> &{ref_suffix} {struct_name} {{
unsafe {{
std::mem::transmute::<*{ptr_suffix} std::ffi::c_void, &{ref_suffix} {struct_name}>(
self.skel_config.map_mmap_ptr{fn_suffix}({idx}).unwrap()
)
}}
}}
}}
"#
)?;
"#
)?;
}
}

Ok(())
Expand Down
22 changes: 17 additions & 5 deletions libbpf-rs/src/skeleton.rs
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ use libbpf_sys::bpf_object_skeleton;
use libbpf_sys::bpf_prog_skeleton;
use libbpf_sys::bpf_program;

use crate::error::IntoError as _;
use crate::libbpf_sys;
use crate::util;
use crate::Error;
Expand Down Expand Up @@ -263,17 +264,28 @@ impl ObjectSkeletonConfig<'_> {
/// `ObjectSkeletonConfigBuilder::map`. Index starts at 0.
///
/// Warning: the returned pointer is only valid while the `ObjectSkeletonConfig` is alive.
pub fn map_mmap_ptr(&mut self, index: usize) -> Result<*mut c_void> {
pub fn map_mmap_ptr(&self, index: usize) -> Result<*const c_void> {
if index >= self.maps.len() {
return Err(Error::with_invalid_data(format!(
"Invalid map index: {index}"
)));
}

self.maps[index].mmaped.as_ref().map_or_else(
|| Err(Error::with_invalid_data("Map does not have mmaped ptr")),
|p| Ok(**p),
)
let p = self.maps[index]
.mmaped
.as_ref()
.ok_or_invalid_data(|| "Map does not have mmaped ptr")?;
Ok(**p)
}

/// Returns the `mmaped` pointer for a map at the specified `index`.
///
/// The index is determined by the order in which the map was passed to
/// `ObjectSkeletonConfigBuilder::map`. Index starts at 0.
///
/// Warning: the returned pointer is only valid while the `ObjectSkeletonConfig` is alive.
pub fn map_mmap_ptr_mut(&mut self, index: usize) -> Result<*mut c_void> {
self.map_mmap_ptr(index).map(|p| p.cast_mut())
}

/// Returns the link pointer for a prog at the specified `index`.
Expand Down

0 comments on commit 16e87b1

Please sign in to comment.