2020-05-28 08:07:52 +00:00
|
|
|
use std::path::PathBuf;
|
2020-06-04 05:48:22 +00:00
|
|
|
use std::collections::{HashMap, HashSet};
|
2020-06-08 07:01:34 +00:00
|
|
|
use std::os::unix::fs::MetadataExt;
|
2020-05-28 08:07:52 +00:00
|
|
|
|
2020-06-03 10:16:08 +00:00
|
|
|
use anyhow::{bail, Error};
|
2020-06-04 05:48:22 +00:00
|
|
|
use lazy_static::lazy_static;
|
|
|
|
|
2020-05-28 08:07:52 +00:00
|
|
|
use super::*;
|
|
|
|
|
2020-06-04 05:48:22 +00:00
|
|
|
lazy_static!{
|
|
|
|
static ref ZFS_UUIDS: HashSet<&'static str> = {
|
|
|
|
let mut set = HashSet::new();
|
|
|
|
set.insert("6a898cc3-1dd2-11b2-99a6-080020736631"); // apple
|
|
|
|
set.insert("516e7cba-6ecf-11d6-8ff8-00022d09712b"); // bsd
|
|
|
|
set
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2020-06-03 10:16:08 +00:00
|
|
|
/// Returns kernel IO-stats for zfs pools
|
2020-05-28 08:07:52 +00:00
|
|
|
pub fn zfs_pool_stats(pool: &OsStr) -> Result<Option<BlockDevStat>, Error> {
|
|
|
|
|
|
|
|
let mut path = PathBuf::from("/proc/spl/kstat/zfs");
|
|
|
|
path.push(pool);
|
|
|
|
path.push("io");
|
|
|
|
|
|
|
|
let text = match proxmox::tools::fs::file_read_optional_string(&path)? {
|
|
|
|
Some(text) => text,
|
|
|
|
None => { return Ok(None); }
|
|
|
|
};
|
|
|
|
|
|
|
|
let lines: Vec<&str> = text.lines().collect();
|
|
|
|
|
|
|
|
if lines.len() < 3 {
|
|
|
|
bail!("unable to parse {:?} - got less than 3 lines", path);
|
|
|
|
}
|
|
|
|
|
|
|
|
// https://github.com/openzfs/zfs/blob/master/lib/libspl/include/sys/kstat.h#L578
|
|
|
|
// nread nwritten reads writes wtime wlentime wupdate rtime rlentime rupdate wcnt rcnt
|
|
|
|
// Note: w -> wait (wtime -> wait time)
|
|
|
|
// Note: r -> run (rtime -> run time)
|
|
|
|
// All times are nanoseconds
|
|
|
|
let stat: Vec<u64> = lines[2].split_ascii_whitespace().map(|s| {
|
|
|
|
u64::from_str_radix(s, 10).unwrap_or(0)
|
|
|
|
}).collect();
|
|
|
|
|
2020-05-28 09:45:34 +00:00
|
|
|
let ticks = (stat[4] + stat[7])/1_000_000; // convert to milisec
|
2020-05-28 08:07:52 +00:00
|
|
|
|
|
|
|
let stat = BlockDevStat {
|
|
|
|
read_sectors: stat[0]>>9,
|
|
|
|
write_sectors: stat[1]>>9,
|
|
|
|
read_ios: stat[2],
|
|
|
|
write_ios: stat[3],
|
2020-05-28 17:11:37 +00:00
|
|
|
io_ticks: ticks,
|
2020-05-28 08:07:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
Ok(Some(stat))
|
|
|
|
}
|
2020-06-03 10:16:08 +00:00
|
|
|
|
2020-06-17 05:00:54 +00:00
|
|
|
|
|
|
|
/// Get set of devices used by zfs (or a specific zfs pool)
|
|
|
|
///
|
|
|
|
/// The set is indexed by using the unix raw device number (dev_t is u64)
|
|
|
|
pub fn zfs_devices(
|
|
|
|
partition_type_map: &HashMap<String, Vec<String>>,
|
|
|
|
pool: Option<String>,
|
|
|
|
) -> Result<HashSet<u64>, Error> {
|
|
|
|
|
|
|
|
let list = zpool_list(pool, true)?;
|
2020-06-03 10:16:08 +00:00
|
|
|
|
2020-06-04 05:48:22 +00:00
|
|
|
let mut device_set = HashSet::new();
|
2020-06-03 10:16:08 +00:00
|
|
|
for entry in list {
|
|
|
|
for device in entry.devices {
|
2020-06-08 07:01:34 +00:00
|
|
|
let meta = std::fs::metadata(device)?;
|
|
|
|
device_set.insert(meta.rdev());
|
2020-06-04 05:48:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for device_list in partition_type_map.iter()
|
|
|
|
.filter_map(|(uuid, list)| if ZFS_UUIDS.contains(uuid.as_str()) { Some(list) } else { None })
|
|
|
|
{
|
|
|
|
for device in device_list {
|
2020-06-08 07:01:34 +00:00
|
|
|
let meta = std::fs::metadata(device)?;
|
|
|
|
device_set.insert(meta.rdev());
|
2020-06-03 10:16:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-04 05:48:22 +00:00
|
|
|
Ok(device_set)
|
2020-06-03 10:16:08 +00:00
|
|
|
}
|
2020-06-16 16:14:35 +00:00
|
|
|
|