1
0
Fork 0
mirror of https://github.com/containers/youki synced 2024-05-06 23:56:13 +02:00
youki/cgroups/src/v2/cpu.rs
Takashi IIGUNI 97848f1ce6
Updated cgroups oci-spec-rs to 0.5.1 or later (#303)
* Updated cgroup oci-spec-rs to 0.5.1

Signed-off-by: Takashi IIGUNI <iiguni.tks@gmail.com>
2021-09-26 18:08:56 -07:00

288 lines
8.3 KiB
Rust

use anyhow::{bail, Context, Result};
use std::path::Path;
use crate::{
common::{self, ControllerOpt},
stats::{CpuUsage, StatsProvider},
};
use oci_spec::runtime::LinuxCpu;
use super::controller::Controller;
const CGROUP_CPU_WEIGHT: &str = "cpu.weight";
const CGROUP_CPU_MAX: &str = "cpu.max";
const DEFAULT_PERIOD: &str = "100000";
const UNRESTRICTED_QUOTA: &str = "max";
const CPU_STAT: &str = "cpu.stat";
pub struct Cpu {}
impl Controller for Cpu {
fn apply(controller_opt: &ControllerOpt, path: &Path) -> Result<()> {
if let Some(cpu) = &controller_opt.resources.cpu() {
Self::apply(path, cpu).context("failed to apply cpu resource restrictions")?;
}
Ok(())
}
}
impl StatsProvider for Cpu {
type Stats = CpuUsage;
fn stats(cgroup_path: &Path) -> Result<Self::Stats> {
let mut stats = CpuUsage::default();
let stat_content = common::read_cgroup_file(cgroup_path.join(CPU_STAT))?;
for entry in stat_content.lines() {
let parts: Vec<&str> = entry.split_ascii_whitespace().collect();
if parts.len() != 2 {
continue;
}
let value = parts[1].parse()?;
match parts[0] {
"usage_usec" => stats.usage_total = value,
"user_usec" => stats.usage_user = value,
"system_usec" => stats.usage_kernel = value,
_ => continue,
}
}
Ok(stats)
}
}
impl Cpu {
fn apply(path: &Path, cpu: &LinuxCpu) -> Result<()> {
if Self::is_realtime_requested(cpu) {
bail!("realtime is not supported on cgroup v2 yet");
}
if let Some(mut shares) = cpu.shares() {
shares = Self::convert_shares_to_cgroup2(shares);
if shares != 0 {
// will result in Erno 34 (numerical result out of range) otherwise
common::write_cgroup_file(path.join(CGROUP_CPU_WEIGHT), shares)?;
}
}
// if quota is unrestricted set to 'max'
let mut quota_string = UNRESTRICTED_QUOTA.to_owned();
if let Some(quota) = cpu.quota() {
if quota > 0 {
quota_string = quota.to_string();
}
}
let mut period_string: String = DEFAULT_PERIOD.to_owned();
if let Some(period) = cpu.period() {
if period > 0 {
period_string = period.to_string();
}
}
// format is 'quota period'
// the kernel default is 'max 100000'
// 250000 250000 -> 1 CPU worth of runtime every 250ms
// 10000 50000 -> 20% of one CPU every 50ms
let max = quota_string + " " + &period_string;
common::write_cgroup_file_str(path.join(CGROUP_CPU_MAX), &max)?;
Ok(())
}
fn convert_shares_to_cgroup2(shares: u64) -> u64 {
if shares == 0 {
return 0;
}
1 + ((shares - 2) * 9999) / 262142
}
fn is_realtime_requested(cpu: &LinuxCpu) -> bool {
if cpu.realtime_period().is_some() {
return true;
}
if cpu.realtime_runtime().is_some() {
return true;
}
false
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::test::{create_temp_dir, set_fixture, setup};
use oci_spec::runtime::LinuxCpuBuilder;
use std::fs;
#[test]
fn test_set_shares() {
// arrange
let (tmp, weight) = setup("test_set_shares", CGROUP_CPU_WEIGHT);
let _ = set_fixture(&tmp, CGROUP_CPU_MAX, "")
.unwrap_or_else(|_| panic!("set test fixture for {}", CGROUP_CPU_MAX));
let cpu = LinuxCpuBuilder::default().shares(22000u64).build().unwrap();
// act
Cpu::apply(&tmp, &cpu).expect("apply cpu");
// assert
let content = fs::read_to_string(weight)
.unwrap_or_else(|_| panic!("read {} file content", CGROUP_CPU_WEIGHT));
assert_eq!(content, 840.to_string());
}
#[test]
fn test_set_positive_quota() {
// arrange
const QUOTA: i64 = 200000;
let (tmp, max) = setup("test_set_positive_quota", CGROUP_CPU_MAX);
let cpu = LinuxCpuBuilder::default().quota(QUOTA).build().unwrap();
// act
Cpu::apply(&tmp, &cpu).expect("apply cpu");
// assert
let content = fs::read_to_string(max)
.unwrap_or_else(|_| panic!("read {} file content", CGROUP_CPU_MAX));
assert_eq!(content, format!("{} {}", QUOTA, DEFAULT_PERIOD))
}
#[test]
fn test_set_zero_quota() {
// arrange
let (tmp, max) = setup("test_set_zero_quota", CGROUP_CPU_MAX);
let cpu = LinuxCpuBuilder::default().quota(0).build().unwrap();
// act
Cpu::apply(&tmp, &cpu).expect("apply cpu");
// assert
let content = fs::read_to_string(max)
.unwrap_or_else(|_| panic!("read {} file content", CGROUP_CPU_MAX));
assert_eq!(
content,
format!("{} {}", UNRESTRICTED_QUOTA, DEFAULT_PERIOD)
)
}
#[test]
fn test_set_positive_period() {
// arrange
const PERIOD: u64 = 100000;
let (tmp, max) = setup("test_set_positive_period", CGROUP_CPU_MAX);
let cpu = LinuxCpuBuilder::default().period(PERIOD).build().unwrap();
// act
Cpu::apply(&tmp, &cpu).expect("apply cpu");
// assert
let content = fs::read_to_string(max)
.unwrap_or_else(|_| panic!("read {} file content", CGROUP_CPU_MAX));
assert_eq!(content, format!("{} {}", UNRESTRICTED_QUOTA, PERIOD))
}
#[test]
fn test_set_zero_period() {
// arrange
let (tmp, max) = setup("test_set_zero_period", CGROUP_CPU_MAX);
let cpu = LinuxCpuBuilder::default().period(0u64).build().unwrap();
// act
Cpu::apply(&tmp, &cpu).expect("apply cpu");
// assert
let content = fs::read_to_string(max)
.unwrap_or_else(|_| panic!("read {} file content", CGROUP_CPU_MAX));
assert_eq!(
content,
format!("{} {}", UNRESTRICTED_QUOTA, DEFAULT_PERIOD)
);
}
#[test]
fn test_set_quota_and_period() {
// arrange
const QUOTA: i64 = 200000;
const PERIOD: u64 = 100000;
let (tmp, max) = setup("test_set_quota_and_period", CGROUP_CPU_MAX);
let cpu = LinuxCpuBuilder::default()
.quota(QUOTA)
.period(PERIOD)
.build()
.unwrap();
// act
Cpu::apply(&tmp, &cpu).expect("apply cpu");
// assert
let content = fs::read_to_string(max)
.unwrap_or_else(|_| panic!("read {} file content", CGROUP_CPU_MAX));
assert_eq!(content, format!("{} {}", QUOTA, PERIOD));
}
#[test]
fn test_realtime_runtime_not_supported() {
// arrange
let tmp = create_temp_dir("test_realtime_runtime_not_supported")
.expect("create temp directory for test");
let cpu = LinuxCpuBuilder::default()
.realtime_runtime(5)
.build()
.unwrap();
// act
let result = Cpu::apply(&tmp, &cpu);
// assert
assert!(
result.is_err(),
"realtime runtime is not supported and should return an error"
);
}
#[test]
fn test_realtime_period_not_supported() {
// arrange
let tmp = create_temp_dir("test_realtime_period_not_supported")
.expect("create temp directory for test");
let cpu = LinuxCpuBuilder::default()
.realtime_period(5u64)
.build()
.unwrap();
// act
let result = Cpu::apply(&tmp, &cpu);
// assert
assert!(
result.is_err(),
"realtime period is not supported and should return an error"
);
}
#[test]
fn test_stat_usage() {
let tmp = create_temp_dir("test_stat_usage").expect("create temp directory for test");
let content = ["usage_usec 7730", "user_usec 4387", "system_usec 3498"].join("\n");
set_fixture(&tmp, CPU_STAT, &content).expect("create stat file");
let actual = Cpu::stats(&tmp).expect("get cgroup stats");
let expected = CpuUsage {
usage_total: 7730,
usage_user: 4387,
usage_kernel: 3498,
..Default::default()
};
assert_eq!(actual, expected);
}
}