From 0e867572ff8e009bbcd1a63037b4ab5b80e35549 Mon Sep 17 00:00:00 2001 From: Tamir Duberstein Date: Tue, 1 Oct 2024 10:13:50 -0400 Subject: [PATCH] Avoid intermediate allocations in parse_cpu_ranges --- aya/src/util.rs | 41 ++++++++++++++++++++++------------------- 1 file changed, 22 insertions(+), 19 deletions(-) diff --git a/aya/src/util.rs b/aya/src/util.rs index d60309b2..b4323163 100644 --- a/aya/src/util.rs +++ b/aya/src/util.rs @@ -203,30 +203,33 @@ fn read_cpu_ranges(path: &'static str) -> Result, (&'static str, io::Er (|| { let data = fs::read_to_string(path)?; parse_cpu_ranges(data.trim()) - .map_err(|range| io::Error::new(io::ErrorKind::InvalidData, range)) })() .map_err(|error| (path, error)) } -fn parse_cpu_ranges(data: &str) -> Result, &str> { - let mut cpus = Vec::new(); - for range in data.split(',') { - cpus.extend({ - match range - .splitn(2, '-') - .map(u32::from_str) - .collect::, _>>() - .map_err(|ParseIntError { .. }| range)? - .as_slice() - { - &[] | &[_, _, _, ..] => return Err(range), - &[start] => start..=start, - &[start, end] => start..=end, - } +fn parse_cpu_ranges(data: &str) -> Result, io::Error> { + data.split(',') + .map(|range| { + let mut iter = range + .split('-') + .map(|s| s.parse::().map_err(|ParseIntError { .. }| range)); + let start = iter.next().unwrap()?; // str::split always returns at least one element. + let end = match iter.next() { + None => start, + Some(end) => { + if iter.next().is_some() { + return Err(range); + } + end? + } + }; + Ok(start..=end) + }) + .try_fold(Vec::new(), |mut cpus, range| { + let range = range.map_err(|range| io::Error::new(io::ErrorKind::InvalidData, range))?; + cpus.extend(range); + Ok(cpus) }) - } - - Ok(cpus) } /// Loads kernel symbols from `/proc/kallsyms`.