2019-04-03 11:39:20 +00:00
|
|
|
use failure::*;
|
|
|
|
|
|
|
|
use crate::tools;
|
|
|
|
use lazy_static::lazy_static;
|
|
|
|
use regex::Regex;
|
2019-04-04 15:08:30 +00:00
|
|
|
use libc;
|
|
|
|
|
|
|
|
/// POSIX sysconf call
|
|
|
|
pub fn sysconf(name: i32) -> i64 {
|
|
|
|
extern { fn sysconf(name: i32) -> i64; }
|
|
|
|
unsafe { sysconf(name) }
|
|
|
|
}
|
|
|
|
|
|
|
|
lazy_static! {
|
|
|
|
static ref CLOCK_TICKS: f64 = sysconf(libc::_SC_CLK_TCK) as f64;
|
|
|
|
}
|
2019-04-03 11:39:20 +00:00
|
|
|
|
|
|
|
pub struct ProcFsPidStat {
|
|
|
|
pub status: u8,
|
|
|
|
pub utime: u64,
|
|
|
|
pub stime: u64,
|
|
|
|
pub starttime: u64,
|
|
|
|
pub vsize: u64,
|
|
|
|
pub rss: i64,
|
|
|
|
}
|
|
|
|
|
2019-04-04 06:05:43 +00:00
|
|
|
pub fn read_proc_pid_stat(pid: libc::pid_t) -> Result<ProcFsPidStat, Error> {
|
2019-04-03 11:39:20 +00:00
|
|
|
|
|
|
|
let statstr = tools::file_read_firstline(format!("/proc/{}/stat", pid))?;
|
|
|
|
|
|
|
|
lazy_static! {
|
2019-04-04 07:25:19 +00:00
|
|
|
static ref REGEX: Regex = Regex::new(concat!(
|
|
|
|
r"^(?P<pid>\d+) \(.*\) (?P<status>\S) -?\d+ -?\d+ -?\d+ -?\d+ -?\d+ \d+ \d+ \d+ \d+ \d+ ",
|
|
|
|
r"(?P<utime>\d+) (?P<stime>\d+) -?\d+ -?\d+ -?\d+ -?\d+ -?\d+ 0 ",
|
|
|
|
r"(?P<starttime>\d+) (?P<vsize>\d+) (?P<rss>-?\d+) ",
|
|
|
|
r"\d+ \d+ \d+ \d+ \d+ \d+ \d+ \d+ \d+ \d+ \d+ \d+ \d+ -?\d+ -?\d+ \d+ \d+ \d+"
|
|
|
|
)).unwrap();
|
2019-04-03 11:39:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(cap) = REGEX.captures(&statstr) {
|
2019-04-04 06:05:43 +00:00
|
|
|
if pid != cap["pid"].parse::<i32>().unwrap() {
|
2019-04-03 11:39:20 +00:00
|
|
|
bail!("unable to read pid stat for process '{}' - got wrong pid", pid);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Ok(ProcFsPidStat {
|
|
|
|
status: cap["status"].as_bytes()[0],
|
|
|
|
utime: cap["utime"].parse::<u64>().unwrap(),
|
|
|
|
stime: cap["stime"].parse::<u64>().unwrap(),
|
|
|
|
starttime: cap["starttime"].parse::<u64>().unwrap(),
|
|
|
|
vsize: cap["vsize"].parse::<u64>().unwrap(),
|
|
|
|
rss: cap["rss"].parse::<i64>().unwrap() * 4096,
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
bail!("unable to read pid stat for process '{}'", pid);
|
|
|
|
}
|
|
|
|
|
2019-04-04 06:05:43 +00:00
|
|
|
pub fn read_proc_starttime(pid: libc::pid_t) -> Result<u64, Error> {
|
2019-04-03 11:39:20 +00:00
|
|
|
|
|
|
|
let info = read_proc_pid_stat(pid)?;
|
|
|
|
|
|
|
|
Ok(info.starttime)
|
|
|
|
}
|
2019-04-04 11:28:14 +00:00
|
|
|
|
|
|
|
pub fn check_process_running(pid: libc::pid_t) -> Option<ProcFsPidStat> {
|
|
|
|
if let Ok(info) = read_proc_pid_stat(pid) {
|
|
|
|
if info.status != 'Z' as u8 {
|
|
|
|
return Some(info);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn check_process_running_pstart(pid: libc::pid_t, pstart: u64) -> Option<ProcFsPidStat> {
|
|
|
|
if let Some(info) = check_process_running(pid) {
|
|
|
|
if info.starttime == pstart {
|
|
|
|
return Some(info);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None
|
|
|
|
}
|
2019-04-04 15:08:30 +00:00
|
|
|
|
|
|
|
pub fn read_proc_uptime() -> Result<(f64, f64), Error> {
|
|
|
|
let file = "/proc/uptime";
|
|
|
|
let line = tools::file_read_firstline(&file)?;
|
|
|
|
let mut values = line.split_whitespace().map(|v| v.parse::<f64>());
|
|
|
|
|
|
|
|
match (values.next(), values.next()) {
|
|
|
|
(Some(Ok(up)), Some(Ok(idle))) => return Ok((up, idle)),
|
|
|
|
_ => bail!("Error while parsing '{}'", file),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn read_proc_uptime_ticks() -> Result<(u64, u64), Error> {
|
|
|
|
let (mut up, mut idle) = read_proc_uptime()?;
|
|
|
|
up *= *CLOCK_TICKS;
|
|
|
|
idle *= *CLOCK_TICKS;
|
|
|
|
Ok((up as u64, idle as u64))
|
|
|
|
}
|