| use libc::SI_LOAD_SHIFT; |
| use std::time::Duration; |
| use std::{cmp, mem}; |
| |
| use crate::errno::Errno; |
| use crate::Result; |
| |
| /// System info structure returned by `sysinfo`. |
| #[derive(Copy, Clone, Debug, Eq, Hash, PartialEq)] |
| #[repr(transparent)] |
| pub struct SysInfo(libc::sysinfo); |
| |
| // The fields are c_ulong on 32-bit linux, u64 on 64-bit linux; x32's ulong is u32 |
| #[cfg(all(target_arch = "x86_64", target_pointer_width = "32"))] |
| type mem_blocks_t = u64; |
| #[cfg(not(all(target_arch = "x86_64", target_pointer_width = "32")))] |
| type mem_blocks_t = libc::c_ulong; |
| |
| impl SysInfo { |
| /// Returns the load average tuple. |
| /// |
| /// The returned values represent the load average over time intervals of |
| /// 1, 5, and 15 minutes, respectively. |
| pub fn load_average(&self) -> (f64, f64, f64) { |
| ( |
| self.0.loads[0] as f64 / (1 << SI_LOAD_SHIFT) as f64, |
| self.0.loads[1] as f64 / (1 << SI_LOAD_SHIFT) as f64, |
| self.0.loads[2] as f64 / (1 << SI_LOAD_SHIFT) as f64, |
| ) |
| } |
| |
| /// Returns the time since system boot. |
| // The cast is not unnecessary on all platforms. |
| #[allow(clippy::unnecessary_cast)] |
| pub fn uptime(&self) -> Duration { |
| // Truncate negative values to 0 |
| Duration::from_secs(cmp::max(self.0.uptime, 0) as u64) |
| } |
| |
| /// Current number of processes. |
| pub fn process_count(&self) -> u16 { |
| self.0.procs |
| } |
| |
| /// Returns the amount of swap memory in Bytes. |
| pub fn swap_total(&self) -> u64 { |
| self.scale_mem(self.0.totalswap) |
| } |
| |
| /// Returns the amount of unused swap memory in Bytes. |
| pub fn swap_free(&self) -> u64 { |
| self.scale_mem(self.0.freeswap) |
| } |
| |
| /// Returns the total amount of installed RAM in Bytes. |
| pub fn ram_total(&self) -> u64 { |
| self.scale_mem(self.0.totalram) |
| } |
| |
| /// Returns the amount of completely unused RAM in Bytes. |
| /// |
| /// "Unused" in this context means that the RAM in neither actively used by |
| /// programs, nor by the operating system as disk cache or buffer. It is |
| /// "wasted" RAM since it currently serves no purpose. |
| pub fn ram_unused(&self) -> u64 { |
| self.scale_mem(self.0.freeram) |
| } |
| |
| // The cast is not unnecessary on all platforms. |
| #[allow(clippy::unnecessary_cast)] |
| fn scale_mem(&self, units: mem_blocks_t) -> u64 { |
| units as u64 * self.0.mem_unit as u64 |
| } |
| } |
| |
| /// Returns system information. |
| /// |
| /// [See `sysinfo(2)`](https://man7.org/linux/man-pages/man2/sysinfo.2.html). |
| pub fn sysinfo() -> Result<SysInfo> { |
| let mut info = mem::MaybeUninit::uninit(); |
| let res = unsafe { libc::sysinfo(info.as_mut_ptr()) }; |
| Errno::result(res).map(|_| unsafe { SysInfo(info.assume_init()) }) |
| } |