mirror of https://github.com/aya-rs/aya
				
				
				
			bpf, perf_map: make maps usable from multiple threads
Change PerfMap API so that individual buffers can be read from multiple threads. Change the way maps are stored in the `Bpf` struct from RefCell to a custom RwLock.pull/1/head
							parent
							
								
									d7c91efb2d
								
							
						
					
					
						commit
						d4e282535b
					
				| @ -0,0 +1,79 @@ | ||||
| use parking_lot::{RwLock, RwLockReadGuard, RwLockWriteGuard}; | ||||
| use std::{ | ||||
|     mem, | ||||
|     ops::{Deref, DerefMut}, | ||||
|     sync::Arc, | ||||
| }; | ||||
| 
 | ||||
| use crate::maps::Map; | ||||
| 
 | ||||
| pub(crate) struct MapLockError; | ||||
| 
 | ||||
| /* FIXME: write a full RwLock implementation that doesn't use borrowing guards
 | ||||
|  * so that try_read() and try_write() don't have to use the ugly lifetime | ||||
|  * extension hack */ | ||||
| 
 | ||||
| #[derive(Debug)] | ||||
| pub(crate) struct MapLock { | ||||
|     inner: Arc<RwLock<Map>>, | ||||
| } | ||||
| 
 | ||||
| impl MapLock { | ||||
|     pub(crate) fn new(map: Map) -> MapLock { | ||||
|         MapLock { | ||||
|             inner: Arc::new(RwLock::new(map)), | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     pub(crate) fn try_read(&self) -> Result<MapRef, MapLockError> { | ||||
|         let lock: Option<RwLockReadGuard<'static, Map>> = | ||||
|             unsafe { mem::transmute(self.inner.try_read()) }; | ||||
|         lock.map(|guard| MapRef { | ||||
|             _lock: self.inner.clone(), | ||||
|             guard, | ||||
|         }) | ||||
|         .ok_or(MapLockError) | ||||
|     } | ||||
| 
 | ||||
|     pub(crate) fn try_write(&self) -> Result<MapRefMut, MapLockError> { | ||||
|         let lock: Option<RwLockWriteGuard<'static, Map>> = | ||||
|             unsafe { mem::transmute(self.inner.try_write()) }; | ||||
|         lock.map(|guard| MapRefMut { | ||||
|             _lock: self.inner.clone(), | ||||
|             guard, | ||||
|         }) | ||||
|         .ok_or(MapLockError) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| pub struct MapRef { | ||||
|     _lock: Arc<RwLock<Map>>, | ||||
|     guard: RwLockReadGuard<'static, Map>, | ||||
| } | ||||
| 
 | ||||
| pub struct MapRefMut { | ||||
|     _lock: Arc<RwLock<Map>>, | ||||
|     guard: RwLockWriteGuard<'static, Map>, | ||||
| } | ||||
| 
 | ||||
| impl Deref for MapRef { | ||||
|     type Target = Map; | ||||
| 
 | ||||
|     fn deref(&self) -> &Map { | ||||
|         &*self.guard | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl Deref for MapRefMut { | ||||
|     type Target = Map; | ||||
| 
 | ||||
|     fn deref(&self) -> &Map { | ||||
|         &*self.guard | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl DerefMut for MapRefMut { | ||||
|     fn deref_mut(&mut self) -> &mut Map { | ||||
|         &mut *self.guard | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,117 @@ | ||||
| use std::{ | ||||
|     convert::TryFrom, | ||||
|     io, | ||||
|     ops::DerefMut, | ||||
|     os::unix::io::{AsRawFd, RawFd}, | ||||
|     sync::Arc, | ||||
| }; | ||||
| 
 | ||||
| use bytes::BytesMut; | ||||
| use libc::{sysconf, _SC_PAGESIZE}; | ||||
| use thiserror::Error; | ||||
| 
 | ||||
| use crate::{ | ||||
|     generated::bpf_map_type::BPF_MAP_TYPE_PERF_EVENT_ARRAY, | ||||
|     maps::{ | ||||
|         perf_map::{Events, PerfBuffer, PerfBufferError}, | ||||
|         Map, MapError, MapRefMut, | ||||
|     }, | ||||
|     sys::bpf_map_update_elem, | ||||
| }; | ||||
| 
 | ||||
| #[derive(Error, Debug)] | ||||
| pub enum PerfMapError { | ||||
|     #[error("error parsing /sys/devices/system/cpu/online")] | ||||
|     InvalidOnlineCpuFile, | ||||
| 
 | ||||
|     #[error("no CPUs specified")] | ||||
|     NoCpus, | ||||
| 
 | ||||
|     #[error("invalid cpu {cpu_id}")] | ||||
|     InvalidCpu { cpu_id: u32 }, | ||||
| 
 | ||||
|     #[error("map error: {0}")] | ||||
|     MapError(#[from] MapError), | ||||
| 
 | ||||
|     #[error("perf buffer error: {0}")] | ||||
|     PerfBufferError(#[from] PerfBufferError), | ||||
| 
 | ||||
|     #[error(transparent)] | ||||
|     IOError(#[from] io::Error), | ||||
| 
 | ||||
|     #[error("bpf_map_update_elem failed: {io_error}")] | ||||
|     UpdateElementError { | ||||
|         #[source] | ||||
|         io_error: io::Error, | ||||
|     }, | ||||
| } | ||||
| 
 | ||||
| pub struct PerfMapBuffer<T: DerefMut<Target = Map>> { | ||||
|     _map: Arc<T>, | ||||
|     buf: PerfBuffer, | ||||
| } | ||||
| 
 | ||||
| impl<T: DerefMut<Target = Map>> PerfMapBuffer<T> { | ||||
|     pub fn readable(&self) -> bool { | ||||
|         self.buf.readable() | ||||
|     } | ||||
| 
 | ||||
|     pub fn read_events(&mut self, buffers: &mut [BytesMut]) -> Result<Events, PerfBufferError> { | ||||
|         self.buf.read_events(buffers) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<T: DerefMut<Target = Map>> AsRawFd for PerfMapBuffer<T> { | ||||
|     fn as_raw_fd(&self) -> RawFd { | ||||
|         self.buf.as_raw_fd() | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| pub struct PerfMap<T: DerefMut<Target = Map>> { | ||||
|     map: Arc<T>, | ||||
|     page_size: usize, | ||||
| } | ||||
| 
 | ||||
| impl<T: DerefMut<Target = Map>> PerfMap<T> { | ||||
|     pub fn new(map: T) -> Result<PerfMap<T>, PerfMapError> { | ||||
|         let map_type = map.obj.def.map_type; | ||||
|         if map_type != BPF_MAP_TYPE_PERF_EVENT_ARRAY as u32 { | ||||
|             return Err(MapError::InvalidMapType { | ||||
|                 map_type: map_type as u32, | ||||
|             })?; | ||||
|         } | ||||
|         let _fd = map.fd_or_err()?; | ||||
| 
 | ||||
|         Ok(PerfMap { | ||||
|             map: Arc::new(map), | ||||
|             // Safety: libc
 | ||||
|             page_size: unsafe { sysconf(_SC_PAGESIZE) } as usize, | ||||
|         }) | ||||
|     } | ||||
| 
 | ||||
|     pub fn open( | ||||
|         &mut self, | ||||
|         index: u32, | ||||
|         page_count: Option<usize>, | ||||
|     ) -> Result<PerfMapBuffer<T>, PerfMapError> { | ||||
|         // FIXME: keep track of open buffers
 | ||||
| 
 | ||||
|         let map_fd = self.map.fd_or_err()?; | ||||
|         let buf = PerfBuffer::open(index, self.page_size, page_count.unwrap_or(2))?; | ||||
|         bpf_map_update_elem(map_fd, &index, &buf.as_raw_fd(), 0) | ||||
|             .map_err(|(_, io_error)| PerfMapError::UpdateElementError { io_error })?; | ||||
| 
 | ||||
|         Ok(PerfMapBuffer { | ||||
|             buf, | ||||
|             _map: self.map.clone(), | ||||
|         }) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl TryFrom<MapRefMut> for PerfMap<MapRefMut> { | ||||
|     type Error = PerfMapError; | ||||
| 
 | ||||
|     fn try_from(a: MapRefMut) -> Result<PerfMap<MapRefMut>, PerfMapError> { | ||||
|         PerfMap::new(a) | ||||
|     } | ||||
| } | ||||
| @ -0,0 +1,54 @@ | ||||
| use std::{fs, io, str::FromStr}; | ||||
| 
 | ||||
| const ONLINE_CPUS: &str = "/sys/devices/system/cpu/online"; | ||||
| 
 | ||||
| pub fn online_cpus() -> Result<Vec<u32>, io::Error> { | ||||
|     let data = fs::read_to_string(ONLINE_CPUS)?; | ||||
|     parse_online_cpus(data.trim()).map_err(|_| { | ||||
|         io::Error::new( | ||||
|             io::ErrorKind::Other, | ||||
|             format!("unexpected {} format", ONLINE_CPUS), | ||||
|         ) | ||||
|     }) | ||||
| } | ||||
| 
 | ||||
| fn parse_online_cpus(data: &str) -> Result<Vec<u32>, ()> { | ||||
|     let mut cpus = Vec::new(); | ||||
|     for range in data.split(',') { | ||||
|         cpus.extend({ | ||||
|             match range | ||||
|                 .splitn(2, '-') | ||||
|                 .map(u32::from_str) | ||||
|                 .collect::<Result<Vec<_>, _>>() | ||||
|                 .map_err(|_| ())? | ||||
|                 .as_slice() | ||||
|             { | ||||
|                 &[] | &[_, _, _, ..] => return Err(()), | ||||
|                 &[start] => start..=start, | ||||
|                 &[start, end] => start..=end, | ||||
|             } | ||||
|         }) | ||||
|     } | ||||
| 
 | ||||
|     Ok(cpus) | ||||
| } | ||||
| 
 | ||||
| #[cfg(test)] | ||||
| mod tests { | ||||
|     use std::iter::FromIterator; | ||||
| 
 | ||||
|     use super::*; | ||||
| 
 | ||||
|     #[test] | ||||
|     fn test_parse_online_cpus() { | ||||
|         assert_eq!(parse_online_cpus("0").unwrap(), vec![0]); | ||||
|         assert_eq!(parse_online_cpus("0,1").unwrap(), vec![0, 1]); | ||||
|         assert_eq!(parse_online_cpus("0,1,2").unwrap(), vec![0, 1, 2]); | ||||
|         assert_eq!(parse_online_cpus("0-7").unwrap(), Vec::from_iter(0..=7)); | ||||
|         assert_eq!(parse_online_cpus("0-3,4-7").unwrap(), Vec::from_iter(0..=7)); | ||||
|         assert_eq!(parse_online_cpus("0-5,6,7").unwrap(), Vec::from_iter(0..=7)); | ||||
|         assert!(parse_online_cpus("").is_err()); | ||||
|         assert!(parse_online_cpus("0-1,2-").is_err()); | ||||
|         assert!(parse_online_cpus("foo").is_err()); | ||||
|     } | ||||
| } | ||||
					Loading…
					
					
				
		Reference in New Issue