1 В избранное 0 Ответвления 0

OSCHINA-MIRROR/openeuler-stratovirt

Присоединиться к Gitlife
Откройте для себя и примите участие в публичных проектах с открытым исходным кодом с участием более 10 миллионов разработчиков. Приватные репозитории также полностью бесплатны :)
Присоединиться бесплатно
В этом репозитории не указан файл с открытой лицензией (LICENSE). При использовании обратитесь к конкретному описанию проекта и его зависимостям в коде.
Клонировать/Скачать
numa.rs 12 КБ
Копировать Редактировать Исходные данные Просмотреть построчно История
MJY-HUST Отправлено 3 лет назад 8a3011b
// Copyright (c) 2022 Huawei Technologies Co.,Ltd. All rights reserved.
//
// StratoVirt is licensed under Mulan PSL v2.
// You can use this software according to the terms and conditions of the Mulan
// PSL v2.
// You may obtain a copy of Mulan PSL v2 at:
// http://license.coscl.org.cn/MulanPSL2
// THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY
// KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO
// NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
// See the Mulan PSL v2 for more details.
use std::cmp::max;
use std::collections::{BTreeMap, HashSet};
use anyhow::{anyhow, bail, Result};
use super::error::ConfigError;
use crate::config::{CmdParser, IntegerList, VmConfig, MAX_NODES};
const MIN_NUMA_DISTANCE: u8 = 10;
#[derive(Default, Debug)]
pub struct NumaDistance {
pub destination: u32,
pub distance: u8,
}
#[derive(Default, Debug)]
pub struct NumaConfig {
pub numa_id: u32,
pub cpus: Vec<u8>,
pub distances: Option<Vec<NumaDistance>>,
pub size: u64,
pub mem_dev: String,
}
#[derive(Default)]
pub struct NumaNode {
pub cpus: Vec<u8>,
pub distances: BTreeMap<u32, u8>,
pub size: u64,
}
pub type NumaNodes = BTreeMap<u32, NumaNode>;
/// Complete the NUMA node parameters from user.
///
/// # Arguments
///
/// * `numa_nodes` - The NUMA node information parsing from user.
/// * `nr_cpus` - The VM cpus number.
/// * `mem_size` - The VM memory size.
pub fn complete_numa_node(numa_nodes: &mut NumaNodes, nr_cpus: u8, mem_size: u64) -> Result<()> {
if numa_nodes.len() > 8 {
bail!(
"NUMA nodes should be less than or equal to 8, now is {}",
numa_nodes.len()
);
}
let mut total_ram_size = 0_u64;
let mut max_cpu_id = 0_u8;
let mut cpus_id = HashSet::<u8>::new();
for (_, node) in numa_nodes.iter() {
total_ram_size += node.size;
for id in node.cpus.iter() {
if cpus_id.contains(id) {
bail!("CPU id {} is repeat, please check it again", *id);
}
cpus_id.insert(*id);
max_cpu_id = max(max_cpu_id, *id);
}
}
if cpus_id.len() < nr_cpus as usize {
if let Some(node_0) = numa_nodes.get_mut(&0) {
for id in 0..nr_cpus {
if !cpus_id.contains(&id) {
node_0.cpus.push(id);
}
}
}
}
if total_ram_size != mem_size {
bail!(
"Total memory {} of NUMA nodes is not equals to memory size {}",
total_ram_size,
mem_size,
);
}
if max_cpu_id >= nr_cpus {
bail!(
"CPU index {} should be smaller than max cpu {}",
max_cpu_id,
nr_cpus
);
}
if cpus_id.len() > nr_cpus as usize {
bail!(
"Total cpu numbers {} of NUMA nodes should be less than or equals to smp {}",
cpus_id.len(),
nr_cpus
);
}
Ok(())
}
/// Parse the NUMA node memory parameters.
///
/// # Arguments
///
/// * `numa_config` - The NUMA node configuration.
pub fn parse_numa_mem(numa_config: &str) -> Result<NumaConfig> {
let mut cmd_parser = CmdParser::new("numa");
cmd_parser
.push("")
.push("nodeid")
.push("cpus")
.push("memdev");
cmd_parser.parse(numa_config)?;
let mut config: NumaConfig = NumaConfig::default();
if let Some(node_id) = cmd_parser.get_value::<u32>("nodeid")? {
if node_id >= MAX_NODES {
return Err(anyhow!(ConfigError::IllegalValue(
"nodeid".to_string(),
0,
true,
MAX_NODES as u64,
false,
)));
}
config.numa_id = node_id;
} else {
return Err(anyhow!(ConfigError::FieldIsMissing("nodeid", "numa")));
}
if let Some(mut cpus) = cmd_parser
.get_value::<IntegerList>("cpus")
.map_err(|_| {
anyhow!(ConfigError::ConvertValueFailed(
String::from("u8"),
"cpus".to_string()
))
})?
.map(|v| v.0.iter().map(|e| *e as u8).collect::<Vec<u8>>())
{
cpus.sort_unstable();
config.cpus = cpus;
} else {
return Err(anyhow!(ConfigError::FieldIsMissing("cpus", "numa")));
}
if let Some(mem_dev) = cmd_parser.get_value::<String>("memdev")? {
config.mem_dev = mem_dev;
} else {
return Err(anyhow!(ConfigError::FieldIsMissing("memdev", "numa")));
}
Ok(config)
}
/// Parse the NUMA node distance parameters.
///
/// # Arguments
///
/// * `numa_dist` - The NUMA node distance configuration.
pub fn parse_numa_distance(numa_dist: &str) -> Result<(u32, NumaDistance)> {
let mut cmd_parser = CmdParser::new("numa");
cmd_parser.push("").push("src").push("dst").push("val");
cmd_parser.parse(numa_dist)?;
let mut dist: NumaDistance = NumaDistance::default();
let numa_id = if let Some(src) = cmd_parser.get_value::<u32>("src")? {
if src >= MAX_NODES {
return Err(anyhow!(ConfigError::IllegalValue(
"src".to_string(),
0,
true,
MAX_NODES as u64,
false,
)));
}
src
} else {
return Err(anyhow!(ConfigError::FieldIsMissing("src", "numa")));
};
if let Some(dst) = cmd_parser.get_value::<u32>("dst")? {
if dst >= MAX_NODES {
return Err(anyhow!(ConfigError::IllegalValue(
"dst".to_string(),
0,
true,
MAX_NODES as u64,
false,
)));
}
dist.destination = dst;
} else {
return Err(anyhow!(ConfigError::FieldIsMissing("dst", "numa")));
}
if let Some(val) = cmd_parser.get_value::<u8>("val")? {
if val < MIN_NUMA_DISTANCE {
bail!("NUMA distance shouldn't be less than 10");
}
if numa_id == dist.destination && val != MIN_NUMA_DISTANCE {
bail!("Local distance of node {} should be 10.", numa_id);
}
if numa_id != dist.destination && val == MIN_NUMA_DISTANCE {
bail!(
"Remote distance of node {} should be more than 10.",
numa_id
);
}
dist.distance = val;
} else {
return Err(anyhow!(ConfigError::FieldIsMissing("val", "numa")));
}
Ok((numa_id, dist))
}
impl VmConfig {
/// Add the NUMA node config to vm config.
///
/// # Arguments
///
/// * `numa_config` - The NUMA node configuration.
pub fn add_numa(&mut self, numa_config: &str) -> Result<()> {
let mut cmd_params = CmdParser::new("numa");
cmd_params.push("");
cmd_params.get_parameters(numa_config)?;
if let Some(numa_type) = cmd_params.get_value::<String>("")? {
self.numa_nodes.push((numa_type, numa_config.to_string()));
}
Ok(())
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_parse_numa_mem() {
let mut vm_config = VmConfig::default();
assert!(vm_config
.add_numa("-numa node,nodeid=0,cpus=0-1,memdev=mem0")
.is_ok());
assert!(vm_config
.add_numa("-numa node,nodeid=1,cpus=2-1,memdev=mem1")
.is_ok());
assert!(vm_config
.add_numa("-numa node,nodeid=2,memdev=mem2")
.is_ok());
assert!(vm_config.add_numa("-numa node,nodeid=3,cpus=3-4").is_ok());
assert!(vm_config
.add_numa("-numa node,nodeid=0,cpus=[0-1:3-5],memdev=mem0")
.is_ok());
let numa = vm_config.numa_nodes.get(0).unwrap();
let numa_config = parse_numa_mem(numa.1.as_str()).unwrap();
assert_eq!(numa_config.cpus, vec![0, 1]);
assert_eq!(numa_config.mem_dev, "mem0");
let numa = vm_config.numa_nodes.get(1).unwrap();
assert!(parse_numa_mem(numa.1.as_str()).is_err());
let numa = vm_config.numa_nodes.get(2).unwrap();
assert!(parse_numa_mem(numa.1.as_str()).is_err());
let numa = vm_config.numa_nodes.get(3).unwrap();
assert!(parse_numa_mem(numa.1.as_str()).is_err());
let numa = vm_config.numa_nodes.get(4).unwrap();
let numa_config = parse_numa_mem(numa.1.as_str()).unwrap();
assert_eq!(numa_config.cpus, vec![0, 1, 3, 4, 5]);
}
#[test]
fn test_parse_numa_distance() {
let mut vm_config = VmConfig::default();
assert!(vm_config.add_numa("-numa dist,src=0,dst=1,val=15").is_ok());
assert!(vm_config.add_numa("-numa dist,dst=1,val=10").is_ok());
assert!(vm_config.add_numa("-numa dist,src=0,val=10").is_ok());
assert!(vm_config.add_numa("-numa dist,src=0,dst=1").is_ok());
assert!(vm_config.add_numa("-numa dist,src=0,dst=1,val=10").is_ok());
let numa = vm_config.numa_nodes.get(0).unwrap();
let dist = parse_numa_distance(numa.1.as_str()).unwrap();
assert_eq!(dist.0, 0);
assert_eq!(dist.1.destination, 1);
assert_eq!(dist.1.distance, 15);
let numa = vm_config.numa_nodes.get(1).unwrap();
assert!(parse_numa_distance(numa.1.as_str()).is_err());
let numa = vm_config.numa_nodes.get(2).unwrap();
assert!(parse_numa_distance(numa.1.as_str()).is_err());
let numa = vm_config.numa_nodes.get(3).unwrap();
assert!(parse_numa_distance(numa.1.as_str()).is_err());
let numa = vm_config.numa_nodes.get(4).unwrap();
assert!(parse_numa_distance(numa.1.as_str()).is_err());
}
#[test]
fn test_check_numa_nodes() {
let nr_cpus = 4;
let mem_size = 2147483648;
let numa_node1 = NumaNode {
cpus: vec![0, 1],
distances: Default::default(),
size: 1073741824,
};
let numa_node2 = NumaNode {
cpus: vec![2, 3],
distances: Default::default(),
size: 1073741824,
};
let mut numa_nodes = BTreeMap::new();
numa_nodes.insert(0, numa_node1);
numa_nodes.insert(1, numa_node2);
assert!(complete_numa_node(&mut numa_nodes, nr_cpus, mem_size).is_ok());
let numa_node3 = NumaNode {
cpus: vec![2],
distances: Default::default(),
size: 1073741824,
};
numa_nodes.remove(&1);
numa_nodes.insert(2, numa_node3);
assert!(complete_numa_node(&mut numa_nodes, nr_cpus, mem_size).is_ok());
let numa_node4 = NumaNode {
cpus: vec![2, 3, 4],
distances: Default::default(),
size: 1073741824,
};
numa_nodes.remove(&1);
numa_nodes.insert(1, numa_node4);
assert!(complete_numa_node(&mut numa_nodes, nr_cpus, mem_size).is_err());
let numa_node5 = NumaNode {
cpus: vec![3, 4],
distances: Default::default(),
size: 1073741824,
};
numa_nodes.remove(&1);
numa_nodes.insert(1, numa_node5);
assert!(complete_numa_node(&mut numa_nodes, nr_cpus, mem_size).is_err());
let numa_node6 = NumaNode {
cpus: vec![0, 1],
distances: Default::default(),
size: 1073741824,
};
numa_nodes.remove(&1);
numa_nodes.insert(1, numa_node6);
assert!(complete_numa_node(&mut numa_nodes, nr_cpus, mem_size).is_err());
let numa_node7 = NumaNode {
cpus: vec![2, 3],
distances: Default::default(),
size: 2147483648,
};
numa_nodes.remove(&1);
numa_nodes.insert(1, numa_node7);
assert!(complete_numa_node(&mut numa_nodes, nr_cpus, mem_size).is_err());
}
}

Опубликовать ( 0 )

Вы можете оставить комментарий после Вход в систему

1
https://gitlife.ru/oschina-mirror/openeuler-stratovirt.git
git@gitlife.ru:oschina-mirror/openeuler-stratovirt.git
oschina-mirror
openeuler-stratovirt
openeuler-stratovirt
v2.2.0-rc6