From 211e7be6cc72a46f833183322a41127ffb61d085 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=D0=BD=D0=B0=D0=B1?= Date: Thu, 21 Nov 2024 01:08:07 +0100 Subject: [PATCH] Allow "set!var = program" at top level that makes var = eval(system(program)) Closes #197 --- man/index.txt | 1 + man/zram-generator.conf.md | 16 ++- src/config.rs | 113 ++++++++++++++---- tests/02-zstd/etc/systemd/zram-generator.conf | 6 +- tests/10-example/bin/xenstore-read | 2 + tests/test_cases.rs | 14 ++- zram-generator.conf.example | 6 + 7 files changed, 134 insertions(+), 24 deletions(-) create mode 100755 tests/10-example/bin/xenstore-read diff --git a/man/index.txt b/man/index.txt index 1944a2bf..36094efe 100644 --- a/man/index.txt +++ b/man/index.txt @@ -3,6 +3,7 @@ zram-generator.conf(5) zram-generator.conf.5.ronn modprobe(8) https://man7.org/linux/man-pages/man8/modprobe.8.html proc(5) https://man7.org/linux/man-pages/man5/proc.5.html +system(3) https://man7.org/linux/man-pages/man3/system.3.html systemd-detect-virt(1) https://freedesktop.org/software/systemd/man/systemd-detect-virt.html systemd.generator(7) https://freedesktop.org/software/systemd/man/systemd.generator.html diff --git a/man/zram-generator.conf.md b/man/zram-generator.conf.md index bb915ed6..f97f066d 100644 --- a/man/zram-generator.conf.md +++ b/man/zram-generator.conf.md @@ -42,7 +42,7 @@ This option thus has higher priority than the configuration files. ## OPTIONS -Each device is configured independently in its `[zramN]` section, where N is a nonnegative integer. Other sections are ignored. +Each device is configured independently in its `[zramN]` section, where N is a nonnegative integer. The global section may contain [DIRECTIVES]. Other sections are ignored. Devices with the final size of *0* will be discarded. @@ -57,6 +57,7 @@ Devices with the final size of *0* will be discarded. * `zram-size`= Sets the size of the zram device as a function of *MemTotal*, available as the `ram` variable. + Additional variables may be provided by [DIRECTIVES]. Arithmetic operators (^%/\*-+), e, π, SI suffixes, log(), int(), ceil(), floor(), round(), abs(), min(), max(), and trigonometric functions are supported. @@ -66,7 +67,7 @@ Devices with the final size of *0* will be discarded. Sets the maximum resident memory limit of the zram device (or *0* for no limit) as a function of *MemTotal*, available as the `ram` variable. - Same format as *zram-size*. Defaults to *0*. + Same format as `zram-size`. Defaults to *0*. * `compression-algorithm`= @@ -113,6 +114,17 @@ Devices with the final size of *0* will be discarded. Defaults to *discard*. +## DIRECTIVES + +The global section (before any section header) may contain directives in the following form: + +* `set!`*variable*=*program* + + *program* is executed by the shell as-if by system(3), + its standard output stream parsed as an arithmetic expression (like `zram-size`/`zram-resident-limit`), + then the result is remembered into *variable*, + usable in later `set!`s and `zram-size`s/`zram-resident-limit`s. + ## ENVIRONMENT VARIABLES Setting `ZRAM_GENERATOR_ROOT` during parsing will cause */proc/meminfo* to be read from *$ZRAM_GENERATOR_ROOT/proc/meminfo* instead, diff --git a/src/config.rs b/src/config.rs index 28d00748..c61ca80f 100644 --- a/src/config.rs +++ b/src/config.rs @@ -10,7 +10,9 @@ use std::ffi::OsString; use std::fmt; use std::fs; use std::io::{prelude::*, BufReader}; +use std::os::unix::process::ExitStatusExt; use std::path::{Component, Path, PathBuf}; +use std::process::{Command, Stdio}; const DEFAULT_ZRAM_SIZE: &str = "min(ram / 2, 4096)"; const DEFAULT_RESIDENT_LIMIT: &str = "0"; @@ -97,14 +99,14 @@ impl Device { fn process_size( &self, zram_option: &Option<(String, fasteval::ExpressionI, fasteval::Slab)>, - memtotal_mb: f64, + ctx: &mut EvalContext, default_size: f64, label: &str, ) -> Result { Ok((match zram_option { Some(zs) => { zs.1.from(&zs.2.ps) - .eval(&zs.2, &mut RamNs(memtotal_mb)) + .eval(&zs.2, ctx) .with_context(|| format!("{} {}", self.name, label)) .and_then(|f| { if f >= 0. { @@ -119,29 +121,29 @@ impl Device { * 1024.0) as u64) } - fn set_disksize_if_enabled(&mut self, memtotal_mb: u64) -> Result<()> { - if !self.is_enabled(memtotal_mb) { + fn set_disksize_if_enabled(&mut self, ctx: &mut EvalContext) -> Result<()> { + if !self.is_enabled(ctx.memtotal_mb) { return Ok(()); } if self.zram_fraction.is_some() || self.max_zram_size_mb.is_some() { // deprecated path let max_mb = self.max_zram_size_mb.unwrap_or(None).unwrap_or(u64::MAX); - self.disksize = ((self.zram_fraction.unwrap_or(0.5) * memtotal_mb as f64) as u64) + self.disksize = ((self.zram_fraction.unwrap_or(0.5) * ctx.memtotal_mb as f64) as u64) .min(max_mb) * (1024 * 1024); } else { self.disksize = self.process_size( &self.zram_size, - memtotal_mb as f64, - (memtotal_mb as f64 / 2.).min(4096.), // DEFAULT_ZRAM_SIZE + ctx, + (ctx.memtotal_mb as f64 / 2.).min(4096.), // DEFAULT_ZRAM_SIZE "zram-size", )?; } self.mem_limit = self.process_size( &self.zram_resident_limit, - memtotal_mb as f64, + ctx, 0., // DEFAULT_RESIDENT_LIMIT "zram-resident-limit", )?; @@ -196,13 +198,18 @@ impl fmt::Display for OptMB { } } -struct RamNs(f64); -impl fasteval::EvalNamespace for RamNs { +struct EvalContext { + memtotal_mb: u64, + additional: BTreeMap, +} +impl fasteval::EvalNamespace for EvalContext { fn lookup(&mut self, name: &str, args: Vec, _: &mut String) -> Option { - if name == "ram" && args.is_empty() { - Some(self.0) - } else { + if !args.is_empty() { None + } else if name == "ram" { + Some(self.memtotal_mb as f64) + } else { + self.additional.get(name).copied() } } } @@ -223,6 +230,57 @@ pub fn read_all_devices(root: &Path, kernel_override: bool) -> Result Result<()> { + let (op, arg) = if let Some(colon) = k.find('!') { + k.split_at(colon + 1) + } else { + warn!( + "{}: invalid outside-of-section key {}, ignoring.", + path.display(), + k + ); + return Ok(()); + }; + + match op { + "set!" => { + let out = Command::new("/bin/sh") + .args(["-c", "--", val]) + .stdin(Stdio::null()) + .stderr(Stdio::inherit()) + .output() + .with_context(|| format!("{}: {}: {}", path.display(), k, val))?; + let exit = out + .status + .code() + .unwrap_or_else(|| 128 + out.status.signal().unwrap()); + if exit != 0 { + warn!("{}: {} exited {}", k, val, exit); + } + + let expr = String::from_utf8(out.stdout) + .with_context(|| format!("{}: {}: {}", path.display(), k, val))?; + let evalled = fasteval::Parser::new() + .parse(&expr, &mut slab.ps) + .and_then(|p| p.from(&slab.ps).eval(slab, ctx)) + .with_context(|| format!("{}: {}: {}: {}", path.display(), k, val, expr))?; + ctx.additional.insert(arg.to_string(), evalled); + } + _ => warn!( + "{}: unknown outside-of-section operation {}, ignoring.", + path.display(), + op + ), + } + Ok(()) +} + fn read_devices( root: &Path, kernel_override: bool, @@ -235,6 +293,11 @@ fn read_devices( } let mut devices: HashMap = HashMap::new(); + let mut slab = fasteval::Slab::new(); + let mut ctx = EvalContext { + memtotal_mb, + additional: BTreeMap::new(), + }; for (_, path) in fragments { let ini = Ini::load_from_file(&path)?; @@ -242,11 +305,9 @@ fn read_devices( for (sname, props) in ini.iter() { let sname = match sname { None => { - warn!( - "{}: ignoring settings outside of section: {:?}", - path.display(), - props - ); + for (k, v) in props.iter() { + toplevel_line(&path, k, v, &mut slab, &mut ctx)?; + } continue; } Some(sname) if sname.starts_with("zram") && sname[4..].parse::().is_ok() => { @@ -275,7 +336,7 @@ fn read_devices( } for dev in devices.values_mut() { - dev.set_disksize_if_enabled(memtotal_mb)?; + dev.set_disksize_if_enabled(&mut ctx)?; } Ok(devices) @@ -571,7 +632,11 @@ foo=0 parse_line(&mut dev, "zram-size", val).unwrap(); } assert!(dev.is_enabled(memtotal_mb)); - dev.set_disksize_if_enabled(memtotal_mb).unwrap(); + dev.set_disksize_if_enabled(&mut EvalContext { + memtotal_mb, + additional: vec![("two".to_string(), 2.)].into_iter().collect(), + }) + .unwrap(); dev.disksize } @@ -583,6 +648,14 @@ foo=0 ); } + #[test] + fn test_eval_size_expression_with_additional() { + assert_eq!( + dev_with_zram_size_size(Some("0.5 * ram * two"), 100), + 50 * 2 * 1024 * 1024 + ); + } + #[test] fn test_eval_size_expression_500() { assert_eq!( diff --git a/tests/02-zstd/etc/systemd/zram-generator.conf b/tests/02-zstd/etc/systemd/zram-generator.conf index 82eefbe1..4c900833 100644 --- a/tests/02-zstd/etc/systemd/zram-generator.conf +++ b/tests/02-zstd/etc/systemd/zram-generator.conf @@ -1,4 +1,8 @@ +set!top = echo 3 +set!bottom = echo 4 +set!ratio = ! echo top / bottom + [zram0] compression-algorithm = zstd host-memory-limit = 2050 -zram-size = ram * 0.75 +zram-size = ram * ratio diff --git a/tests/10-example/bin/xenstore-read b/tests/10-example/bin/xenstore-read new file mode 100755 index 00000000..c6b08a3e --- /dev/null +++ b/tests/10-example/bin/xenstore-read @@ -0,0 +1,2 @@ +#!/bin/sh -x +echo 2 diff --git a/tests/test_cases.rs b/tests/test_cases.rs index aafdfbdf..095f8c28 100644 --- a/tests/test_cases.rs +++ b/tests/test_cases.rs @@ -4,8 +4,11 @@ use zram_generator::{config, generator}; use anyhow::Result; use fs_extra::dir::{copy, CopyOptions}; +use std::env; +use std::ffi::OsString; use std::fs; use std::io::{self, Write}; +use std::os::unix::ffi::OsStringExt; use std::path::Path; use std::process::{exit, Command}; use tempfile::TempDir; @@ -43,6 +46,13 @@ fn unshorn() { .unwrap(); fs::create_dir("/proc/self").unwrap(); symlink("zram-generator", "/proc/self/exe").unwrap(); + + let mut path = env::var_os("PATH").map(|p| p.to_os_string().into_vec()).unwrap_or(b"/usr/bin:/bin".to_vec()); // _PATH_DEFPATH + path.insert(0, b':'); + for &b in "tests/10-example/bin".as_bytes().into_iter().rev() { + path.insert(0, b); + } + env::set_var("PATH", OsString::from_vec(path)); } fn prepare_directory(srcroot: &Path) -> Result { @@ -65,6 +75,7 @@ fn prepare_directory(srcroot: &Path) -> Result { } fn test_generation(path: &str) -> Result> { + let srcroot = Path::new(path); let rootdir = prepare_directory(&srcroot)?; let root = rootdir.path(); @@ -123,7 +134,8 @@ fn test_02_zstd() { let d = &devices[0]; assert!(d.is_swap()); assert_eq!(d.host_memory_limit_mb, Some(2050)); - assert_eq!(d.zram_size.as_ref().map(z_s_name), Some("ram * 0.75")); + assert_eq!(d.zram_size.as_ref().map(z_s_name), Some("ram * ratio")); + assert_eq!(d.disksize, 614989824); assert_eq!(d.compression_algorithm.as_ref().unwrap(), "zstd"); assert_eq!(d.options, "discard"); } diff --git a/zram-generator.conf.example b/zram-generator.conf.example index 0a9af4ab..7fe89ead 100644 --- a/zram-generator.conf.example +++ b/zram-generator.conf.example @@ -1,6 +1,12 @@ # This file is part of the zram-generator project # https://github.com/systemd/zram-generator +# At the top level, a set!variable = program +# directive executes /bin/sh -c program, +# parses the output as an expression, and remembers it in variable, +# usable in later set! and zram-size/zram-resident-limit. +set!maxhotplug = xenstore-read /local/domain/$(xenstore-read domid)/memory/hotplug-max + [zram0] # This section describes the settings for /dev/zram0. #