From 3079bc5a6080201c5207a1de00db4a1e9d8ee825 Mon Sep 17 00:00:00 2001 From: Guillaume Gomez Date: Tue, 12 Nov 2024 00:07:33 +0100 Subject: [PATCH 1/2] Add disk I/O support on freebsd --- Cargo.toml | 1 + src/unix/freebsd/disk.rs | 297 ++++++++++++++++++++++++++++++++------ src/unix/freebsd/ffi.rs | 75 ++++++++++ src/unix/freebsd/mod.rs | 1 + src/unix/freebsd/utils.rs | 4 +- tests/disk.rs | 4 - 6 files changed, 334 insertions(+), 48 deletions(-) create mode 100644 src/unix/freebsd/ffi.rs diff --git a/Cargo.toml b/Cargo.toml index 14c0a7407..652a6b617 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -120,3 +120,4 @@ tempfile = "3.9" name = "simple" path = "examples/simple.rs" required-features = ["default"] +doc-scrape-examples = true diff --git a/src/unix/freebsd/disk.rs b/src/unix/freebsd/disk.rs index b4194e33f..124f47bf8 100644 --- a/src/unix/freebsd/disk.rs +++ b/src/unix/freebsd/disk.rs @@ -2,21 +2,43 @@ use crate::{Disk, DiskKind, DiskUsage}; +use std::cell::RefCell; +use std::collections::HashMap; use std::ffi::{OsStr, OsString}; use std::os::unix::ffi::OsStringExt; use std::path::{Path, PathBuf}; - -use super::utils::c_buf_to_utf8_str; - +use std::ptr::null_mut; + +use super::ffi::{ + devinfo, + devstat, + devstat_compute_statistics, + devstat_getdevs, + devstat_getversion, + statinfo, + DSM_NONE, + DSM_TOTAL_BYTES_READ, + DSM_TOTAL_BYTES_WRITE, +}; + +use super::utils::{c_buf_to_utf8_str, c_buf_to_utf8_string, get_sys_value_str_by_name}; + +#[derive(Debug)] pub(crate) struct DiskInner { name: OsString, c_mount_point: Vec, + dev_id: Option, mount_point: PathBuf, total_space: u64, available_space: u64, file_system: OsString, is_removable: bool, is_read_only: bool, + read_bytes: u64, + old_read_bytes: u64, + written_bytes: u64, + old_written_bytes: u64, + updated: bool, } impl DiskInner { @@ -53,15 +75,16 @@ impl DiskInner { } pub(crate) fn refresh(&mut self) -> bool { - unsafe { - let mut vfs: libc::statvfs = std::mem::zeroed(); - refresh_disk(self, &mut vfs) - } + refresh_disk(self) } pub(crate) fn usage(&self) -> DiskUsage { - // TODO: Until disk i/o stats are added, return the default - DiskUsage::default() + DiskUsage { + read_bytes: self.read_bytes.saturating_sub(self.old_read_bytes), + total_read_bytes: self.read_bytes, + written_bytes: self.written_bytes.saturating_sub(self.old_written_bytes), + total_written_bytes: self.written_bytes, + } } } @@ -73,7 +96,7 @@ impl crate::DisksInner { } pub(crate) fn refresh_list(&mut self) { - unsafe { get_all_list(&mut self.disks) } + unsafe { get_all_list(&mut self.disks, true); } } pub(crate) fn list(&self) -> &[Disk] { @@ -85,36 +108,192 @@ impl crate::DisksInner { } pub(crate) fn refresh(&mut self) { - for disk in self.list_mut() { - disk.refresh(); + unsafe { get_all_list(&mut self.disks, false); } + } +} + +trait GetValues { + fn update_old(&mut self); + fn get_read(&mut self) -> &mut u64; + fn get_written(&mut self) -> &mut u64; + fn dev_id(&self) -> Option<&String>; +} + +impl GetValues for crate::Disk { + fn update_old(&mut self) { + self.inner.update_old() + } + fn get_read(&mut self) -> &mut u64 { + self.inner.get_read() + } + fn get_written(&mut self) -> &mut u64 { + self.inner.get_written() + } + fn dev_id(&self) -> Option<&String> { + self.inner.dev_id() + } +} + +impl<'a> GetValues for &'a mut DiskInner { + fn update_old(&mut self) { + self.old_read_bytes = self.read_bytes; + self.old_written_bytes = self.written_bytes; + } + fn get_read(&mut self) -> &mut u64 { + &mut self.read_bytes + } + fn get_written(&mut self) -> &mut u64 { + &mut self.written_bytes + } + fn dev_id(&self) -> Option<&String> { + self.dev_id.as_ref() + } +} +impl GetValues for DiskInner { + fn update_old(&mut self) { + self.old_read_bytes = self.read_bytes; + self.old_written_bytes = self.written_bytes; + } + fn get_read(&mut self) -> &mut u64 { + &mut self.read_bytes + } + fn get_written(&mut self) -> &mut u64 { + &mut self.written_bytes + } + fn dev_id(&self) -> Option<&String> { + self.dev_id.as_ref() + } +} + +fn refresh_disk(disk: &mut DiskInner) -> bool { + unsafe { + let mut vfs: libc::statvfs = std::mem::zeroed(); + if libc::statvfs(disk.c_mount_point.as_ptr() as *const _, &mut vfs as *mut _) < 0 { + return false; + } + let block_size: u64 = vfs.f_frsize as _; + + disk.total_space = vfs.f_blocks.saturating_mul(block_size); + disk.available_space = vfs.f_favail.saturating_mul(block_size); + refresh_disk_io(&mut [disk]); + true + } +} + +struct DevInfoWrapper { + info: statinfo, +} + +impl DevInfoWrapper { + fn new() -> Self { + Self { + info: unsafe { std::mem::zeroed() }, + } + } + + unsafe fn get_devs(&mut self) -> Option<&statinfo> { + let version = devstat_getversion(null_mut()); + if version != 6 { + // For now we only handle the devstat 6 version. + sysinfo_debug!("version {version} of devstat is not supported"); + return None; + } + if self.info.dinfo.is_null() { + self.info.dinfo = libc::calloc(1, std::mem::size_of::()) as *mut _; + if self.info.dinfo.is_null() { + return None; + } + } + if devstat_getdevs(null_mut(), &mut self.info as *mut _) != -1 { + Some(&self.info) + } else { + None } } } -// FIXME: if you want to get disk I/O usage: -// statfs.[f_syncwrites, f_asyncwrites, f_syncreads, f_asyncreads] +impl Drop for DevInfoWrapper { + fn drop(&mut self) { + if !self.info.dinfo.is_null() { + unsafe { libc::free(self.info.dinfo as *mut _); } + } + } +} -unsafe fn refresh_disk(disk: &mut DiskInner, vfs: &mut libc::statvfs) -> bool { - if libc::statvfs(disk.c_mount_point.as_ptr() as *const _, vfs) < 0 { - return false; +unsafe fn refresh_disk_io(disks: &mut [T]) { + thread_local! { + static DEV_INFO: RefCell = RefCell::new(DevInfoWrapper::new()); } - let f_frsize: u64 = vfs.f_frsize as _; - disk.total_space = vfs.f_blocks.saturating_mul(f_frsize); - disk.available_space = vfs.f_favail.saturating_mul(f_frsize); - true + DEV_INFO.with_borrow_mut(|dev_info| { + let Some(stat_info) = dev_info.get_devs() else { return }; + let dinfo = (*stat_info).dinfo; + + let numdevs = (*dinfo).numdevs; + if numdevs < 0 { + return; + } + let devices: &mut [devstat] = std::slice::from_raw_parts_mut((*dinfo).devices, numdevs as _); + for device in devices { + let Some(device_name) = c_buf_to_utf8_str(&device.device_name) else { continue }; + let dev_stat_name = format!("{device_name}{}", device.unit_number); + + for disk in disks.iter_mut().filter(|d| d.dev_id().is_some_and(|id| *id == dev_stat_name)) { + disk.update_old(); + let mut read = 0u64; + devstat_compute_statistics( + device, + null_mut(), + 0, + DSM_TOTAL_BYTES_READ, + &mut read, + DSM_TOTAL_BYTES_WRITE, + disk.get_written(), + DSM_NONE, + ); + *disk.get_read() = read; + } + } + }); } -pub unsafe fn get_all_list(container: &mut Vec) { - container.clear(); +fn get_disks_mapping() -> HashMap { + let mut disk_mapping = HashMap::new(); + let Some(mapping) = get_sys_value_str_by_name(b"kern.geom.conftxt\0") else { return disk_mapping }; + + let mut last_id = String::new(); + + for line in mapping.lines() { + let mut parts = line.split_whitespace(); + let Some(kind) = parts.next() else { continue }; + if kind == "0" { + if let Some("DISK") = parts.next() { + if let Some(id) = parts.next() { + last_id.clear(); + last_id.push_str(id); + } + } + } else if kind == "2" && !last_id.is_empty() { + if let Some("LABEL") = parts.next() { + if let Some(path) = parts.next() { + disk_mapping.insert(format!("/dev/{path}"), last_id.clone()); + } + } + } + } + return disk_mapping; +} - let mut fs_infos: *mut libc::statfs = std::ptr::null_mut(); +pub unsafe fn get_all_list(container: &mut Vec, add_new_disks: bool) { + let mut fs_infos: *mut libc::statfs = null_mut(); let count = libc::getmntinfo(&mut fs_infos, libc::MNT_WAIT); if count < 1 { return; } + let disk_mapping = get_disks_mapping(); + let mut vfs: libc::statvfs = std::mem::zeroed(); let fs_infos: &[libc::statfs] = std::slice::from_raw_parts(fs_infos as _, count as _); @@ -146,10 +325,6 @@ pub unsafe fn get_all_list(container: &mut Vec) { _ => {} } - if libc::statvfs(fs_info.f_mntonname.as_ptr(), &mut vfs) != 0 { - continue; - } - let mount_point = match c_buf_to_utf8_str(&fs_info.f_mntonname) { Some(m) => m, None => { @@ -158,31 +333,69 @@ pub unsafe fn get_all_list(container: &mut Vec) { } }; + if mount_point == "/boot/efi" { + continue; + } let name = if mount_point == "/" { OsString::from("root") } else { OsString::from(mount_point) }; - // USB keys and CDs are removable. - let is_removable = - [b"USB", b"usb"].iter().any(|b| *b == &fs_type[..]) || fs_type.starts_with(b"/dev/cd"); + if libc::statvfs(fs_info.f_mntonname.as_ptr(), &mut vfs) != 0 { + continue; + } let f_frsize: u64 = vfs.f_frsize as _; let is_read_only = (vfs.f_flag & libc::ST_RDONLY) != 0; + let total_space = vfs.f_blocks.saturating_mul(f_frsize); + let available_space = vfs.f_favail.saturating_mul(f_frsize); + + if let Some(disk) = container.iter_mut().find(|d| d.inner.name == name) { + disk.inner.updated = true; + disk.inner.total_space = total_space; + disk.inner.available_space = available_space; + } else if add_new_disks { + let dev_mount_point = c_buf_to_utf8_str(&fs_info.f_mntfromname).unwrap_or(""); + + // USB keys and CDs are removable. + let is_removable = + [b"USB", b"usb"].iter().any(|b| *b == &fs_type[..]) || fs_type.starts_with(b"/dev/cd"); + + container.push(Disk { + inner: DiskInner { + name, + c_mount_point: fs_info.f_mntonname.to_vec(), + mount_point: PathBuf::from(mount_point), + dev_id: disk_mapping.get(dev_mount_point).map(ToString::to_string), + total_space: vfs.f_blocks.saturating_mul(f_frsize), + available_space: vfs.f_favail.saturating_mul(f_frsize), + file_system: OsString::from_vec(fs_type), + is_removable, + is_read_only, + read_bytes: 0, + old_read_bytes: 0, + written_bytes: 0, + old_written_bytes: 0, + updated: true, + }, + }); + } + } - container.push(Disk { - inner: DiskInner { - name, - c_mount_point: fs_info.f_mntonname.to_vec(), - mount_point: PathBuf::from(mount_point), - total_space: vfs.f_blocks.saturating_mul(f_frsize), - available_space: vfs.f_favail.saturating_mul(f_frsize), - file_system: OsString::from_vec(fs_type), - is_removable, - is_read_only, - }, + if add_new_disks { + container.retain_mut(|disk| { + if !disk.inner.updated { + return false; + } + disk.inner.updated = false; + true }); + } else { + for c in container { + c.inner.updated = false; + } } + refresh_disk_io(container); } diff --git a/src/unix/freebsd/ffi.rs b/src/unix/freebsd/ffi.rs new file mode 100644 index 000000000..1c2aab2e8 --- /dev/null +++ b/src/unix/freebsd/ffi.rs @@ -0,0 +1,75 @@ +#![allow(non_camel_case_types)] +// Because of long double. +#![allow(improper_ctypes)] + +use libc::{c_char, c_int, c_long, c_uint, c_void, bintime, kvm_t, CPUSTATES}; + +// definitions come from: +// https://github.com/freebsd/freebsd-src/blob/main/lib/libdevstat/devstat.h +// https://github.com/freebsd/freebsd-src/blob/main/sys/sys/devicestat.h + +// 16 bytes says `sizeof` in C so let's use a type of the same size. +pub type c_long_double = u128; +pub type devstat_priority = c_int; +pub type devstat_support_flags = c_int; +pub type devstat_type_flags = c_int; + +#[repr(C)] +pub(crate) struct tailq { + pub(crate) stqe_next: *mut devstat, +} + +#[repr(C)] +pub(crate) struct devstat { + pub(crate) sequence0: c_uint, + pub(crate) allocated: c_int, + pub(crate) start_count: c_uint, + pub(crate) end_count: c_uint, + pub(crate) busy_from: bintime, + pub(crate) dev_links: tailq, + pub(crate) device_number: u32, + pub(crate) device_name: [c_char; DEVSTAT_NAME_LEN], + pub(crate) unit_number: c_int, + pub(crate) bytes: [u64; DEVSTAT_N_TRANS_FLAGS], + pub(crate) operations: [u64; DEVSTAT_N_TRANS_FLAGS], + pub(crate) duration: [bintime; DEVSTAT_N_TRANS_FLAGS], + pub(crate) busy_time: bintime, + pub(crate) creation_time: bintime, + pub(crate) block_size: u32, + pub(crate) tag_types: [u64; 3], + pub(crate) flags: devstat_support_flags, + pub(crate) device_type: devstat_type_flags, + pub(crate) priority: devstat_priority, + pub(crate) id: *const c_void, + pub(crate) sequence1: c_uint, +} + +#[repr(C)] +pub(crate) struct devinfo { + pub(crate) devices: *mut devstat, + pub(crate) mem_ptr: *mut u8, + pub(crate) generation: c_long, + pub(crate) numdevs: c_int, +} + +#[repr(C)] +pub(crate) struct statinfo { + pub(crate) cp_time: [c_long; CPUSTATES as usize], + pub(crate) tk_nin: c_long, + pub(crate) tk_nout: c_long, + pub(crate) dinfo: *mut devinfo, + pub(crate) snap_time: c_long_double, +} + +pub(crate) const DEVSTAT_N_TRANS_FLAGS: usize = 4; +pub(crate) const DEVSTAT_NAME_LEN: usize = 16; + +pub(crate) const DSM_NONE: c_int = 0; +pub(crate) const DSM_TOTAL_BYTES_READ: c_int = 2; +pub(crate) const DSM_TOTAL_BYTES_WRITE: c_int = 3; + +extern "C" { + pub(crate) fn devstat_getversion(kd: *mut kvm_t) -> c_int; + pub(crate) fn devstat_getdevs(kd: *mut kvm_t, stats: *mut statinfo) -> c_int; + pub(crate) fn devstat_compute_statistics(current: *mut devstat, previous: *mut devstat, etime: c_long_double, ...) -> c_int; +} diff --git a/src/unix/freebsd/mod.rs b/src/unix/freebsd/mod.rs index 3dabf51e8..e1a8fdbd3 100644 --- a/src/unix/freebsd/mod.rs +++ b/src/unix/freebsd/mod.rs @@ -15,6 +15,7 @@ cfg_if! { } if #[cfg(feature = "disk")] { pub mod disk; + pub mod ffi; pub(crate) use self::disk::DiskInner; pub(crate) use crate::unix::DisksInner; diff --git a/src/unix/freebsd/utils.rs b/src/unix/freebsd/utils.rs index 6aa611f01..908e91d77 100644 --- a/src/unix/freebsd/utils.rs +++ b/src/unix/freebsd/utils.rs @@ -77,7 +77,7 @@ pub(crate) fn c_buf_to_utf8_str(buf: &[libc::c_char]) -> Option<&str> { } } -#[cfg(any(feature = "system", feature = "network"))] +#[cfg(any(feature = "disk", feature = "system", feature = "network"))] pub(crate) fn c_buf_to_utf8_string(buf: &[libc::c_char]) -> Option { c_buf_to_utf8_str(buf).map(|s| s.to_owned()) } @@ -137,7 +137,7 @@ pub(crate) unsafe fn get_sys_value_by_name(name: &[u8], value: &mut T) && original == len } -#[cfg(feature = "system")] +#[cfg(any(feature = "system", feature = "disk"))] pub(crate) fn get_sys_value_str_by_name(name: &[u8]) -> Option { let mut size = 0; diff --git a/tests/disk.rs b/tests/disk.rs index 4839c7d16..4210fcac7 100644 --- a/tests/disk.rs +++ b/tests/disk.rs @@ -72,9 +72,5 @@ fn test_disks_usage() { // written_bytes should have increased by about 10mb, but this is not fully reliable in CI Linux. For now, // just verify the number is non-zero. - #[cfg(not(target_os = "freebsd"))] assert!(written_bytes > 0); - // Disk usage is not yet supported on freebsd - #[cfg(target_os = "freebsd")] - assert_eq!(written_bytes, 0); } From 48d04ec3ce5f8f410ca3fc7f093536019847ac93 Mon Sep 17 00:00:00 2001 From: Guillaume Gomez Date: Mon, 25 Nov 2024 00:37:34 +0100 Subject: [PATCH 2/2] Use libgeom to get disk I/O information on FreeBSD --- src/common/disk.rs | 2 - src/unix/freebsd/disk.rs | 260 ++++++++++++++++++++++++++------------- src/unix/freebsd/ffi.rs | 59 +++++---- 3 files changed, 210 insertions(+), 111 deletions(-) diff --git a/src/common/disk.rs b/src/common/disk.rs index 0c2712bf2..1fe004f11 100644 --- a/src/common/disk.rs +++ b/src/common/disk.rs @@ -149,8 +149,6 @@ impl Disk { /// Returns number of bytes read and written by the disk /// - /// ⚠️ Note that FreeBSD is not yet supported - /// /// ```no_run /// use sysinfo::Disks; /// diff --git a/src/unix/freebsd/disk.rs b/src/unix/freebsd/disk.rs index 124f47bf8..ec841603e 100644 --- a/src/unix/freebsd/disk.rs +++ b/src/unix/freebsd/disk.rs @@ -1,27 +1,28 @@ // Take a look at the license at the top of the repository in the LICENSE file. -use crate::{Disk, DiskKind, DiskUsage}; - -use std::cell::RefCell; use std::collections::HashMap; use std::ffi::{OsStr, OsString}; +use std::marker::PhantomData; use std::os::unix::ffi::OsStringExt; use std::path::{Path, PathBuf}; -use std::ptr::null_mut; +use std::ptr::{NonNull, null_mut}; +use std::sync::OnceLock; +use libc::c_void; + +use crate::{Disk, DiskKind, DiskUsage}; use super::ffi::{ - devinfo, devstat, - devstat_compute_statistics, - devstat_getdevs, devstat_getversion, - statinfo, - DSM_NONE, - DSM_TOTAL_BYTES_READ, - DSM_TOTAL_BYTES_WRITE, + geom_stats_open, + geom_stats_snapshot_get, + geom_stats_snapshot_next, + geom_stats_snapshot_reset, + geom_stats_snapshot_free, + DEVSTAT_READ, + DEVSTAT_WRITE, }; - -use super::utils::{c_buf_to_utf8_str, c_buf_to_utf8_string, get_sys_value_str_by_name}; +use super::utils::{c_buf_to_utf8_str, get_sys_value_str_by_name}; #[derive(Debug)] pub(crate) struct DiskInner { @@ -134,7 +135,7 @@ impl GetValues for crate::Disk { } } -impl<'a> GetValues for &'a mut DiskInner { +impl GetValues for &mut DiskInner { fn update_old(&mut self) { self.old_read_bytes = self.read_bytes; self.old_written_bytes = self.written_bytes; @@ -180,81 +181,77 @@ fn refresh_disk(disk: &mut DiskInner) -> bool { } } -struct DevInfoWrapper { - info: statinfo, +unsafe fn initialize_geom() -> Result<(), ()> { + let version = devstat_getversion(null_mut()); + if version != 6 { + // For now we only handle the devstat 6 version. + sysinfo_debug!("version {version} of devstat is not supported"); + return Err(()); + } + let r = unsafe { geom_stats_open() }; + if r != 0 { + sysinfo_debug!("`geom_stats_open` failed: {r}"); + Err(()) + } else { + Ok(()) + } } -impl DevInfoWrapper { - fn new() -> Self { - Self { - info: unsafe { std::mem::zeroed() }, - } - } +unsafe fn refresh_disk_io(disks: &mut [T]) { + static GEOM_STATS: OnceLock> = OnceLock::new(); - unsafe fn get_devs(&mut self) -> Option<&statinfo> { - let version = devstat_getversion(null_mut()); - if version != 6 { - // For now we only handle the devstat 6 version. - sysinfo_debug!("version {version} of devstat is not supported"); - return None; - } - if self.info.dinfo.is_null() { - self.info.dinfo = libc::calloc(1, std::mem::size_of::()) as *mut _; - if self.info.dinfo.is_null() { - return None; - } - } - if devstat_getdevs(null_mut(), &mut self.info as *mut _) != -1 { - Some(&self.info) - } else { - None - } + if GEOM_STATS.get_or_init(|| unsafe { initialize_geom() }).is_err() { + return; } -} + let Some(mut snap) = GeomSnapshot::new() else { return }; + for device in snap.iter() { + let device = device.devstat.as_ref(); + let Some(device_name) = c_buf_to_utf8_str(&device.device_name) else { continue }; + let dev_stat_name = format!("{device_name}{}", device.unit_number); -impl Drop for DevInfoWrapper { - fn drop(&mut self) { - if !self.info.dinfo.is_null() { - unsafe { libc::free(self.info.dinfo as *mut _); } + for disk in disks.iter_mut().filter(|d| d.dev_id().is_some_and(|id| *id == dev_stat_name)) { + disk.update_old(); + *disk.get_read() = device.bytes[DEVSTAT_READ]; + *disk.get_written() = device.bytes[DEVSTAT_WRITE]; } } -} -unsafe fn refresh_disk_io(disks: &mut [T]) { - thread_local! { - static DEV_INFO: RefCell = RefCell::new(DevInfoWrapper::new()); - } - - DEV_INFO.with_borrow_mut(|dev_info| { - let Some(stat_info) = dev_info.get_devs() else { return }; - let dinfo = (*stat_info).dinfo; - - let numdevs = (*dinfo).numdevs; - if numdevs < 0 { - return; - } - let devices: &mut [devstat] = std::slice::from_raw_parts_mut((*dinfo).devices, numdevs as _); - for device in devices { - let Some(device_name) = c_buf_to_utf8_str(&device.device_name) else { continue }; - let dev_stat_name = format!("{device_name}{}", device.unit_number); - - for disk in disks.iter_mut().filter(|d| d.dev_id().is_some_and(|id| *id == dev_stat_name)) { - disk.update_old(); - let mut read = 0u64; - devstat_compute_statistics( - device, - null_mut(), - 0, - DSM_TOTAL_BYTES_READ, - &mut read, - DSM_TOTAL_BYTES_WRITE, - disk.get_written(), - DSM_NONE, - ); - *disk.get_read() = read; - } - } - }); + // thread_local! { + // static DEV_INFO: RefCell = RefCell::new(DevInfoWrapper::new()); + // } + + // DEV_INFO.with_borrow_mut(|dev_info| { + // let Some(stat_info) = dev_info.get_devs() else { return }; + // let dinfo = (*stat_info).dinfo; + + // let numdevs = (*dinfo).numdevs; + // if numdevs < 0 { + // return; + // } + // let devices: &mut [devstat] = std::slice::from_raw_parts_mut((*dinfo).devices, numdevs as _); + // for device in devices { + // let Some(device_name) = c_buf_to_utf8_str(&device.device_name) else { continue }; + // let dev_stat_name = format!("{device_name}{}", device.unit_number); + + // for disk in disks.iter_mut().filter(|d| d.dev_id().is_some_and(|id| *id == dev_stat_name)) { + // disk.update_old(); + // let mut read = 0u64; + // // This code cannot work because `devstat_compute_statistics` expects a + // // `long double` as 3rd argument, making it impossible for rust to call it... + // devstat_compute_statistics( + // device, + // null_mut(), + // 0, + // DSM_TOTAL_BYTES_READ, + // &mut read, + // DSM_TOTAL_BYTES_WRITE, + // disk.get_written(), + // DSM_NONE, + // ); + // *disk.get_read() = read; + // } + // } + // }); } fn get_disks_mapping() -> HashMap { @@ -281,7 +278,7 @@ fn get_disks_mapping() -> HashMap { } } } - return disk_mapping; + disk_mapping } pub unsafe fn get_all_list(container: &mut Vec, add_new_disks: bool) { @@ -393,9 +390,104 @@ pub unsafe fn get_all_list(container: &mut Vec, add_new_disks: bool) { true }); } else { - for c in container { + for c in container.iter_mut() { c.inner.updated = false; } } - refresh_disk_io(container); + refresh_disk_io(container.as_mut_slice()); +} + +// struct DevInfoWrapper { +// info: statinfo, +// } + +// impl DevInfoWrapper { +// fn new() -> Self { +// Self { +// info: unsafe { std::mem::zeroed() }, +// } +// } + +// unsafe fn get_devs(&mut self) -> Option<&statinfo> { +// let version = devstat_getversion(null_mut()); +// if version != 6 { +// // For now we only handle the devstat 6 version. +// sysinfo_debug!("version {version} of devstat is not supported"); +// return None; +// } +// if self.info.dinfo.is_null() { +// self.info.dinfo = libc::calloc(1, std::mem::size_of::()) as *mut _; +// if self.info.dinfo.is_null() { +// return None; +// } +// } +// if devstat_getdevs(null_mut(), &mut self.info as *mut _) != -1 { +// Some(&self.info) +// } else { +// None +// } +// } +// } + +// impl Drop for DevInfoWrapper { +// fn drop(&mut self) { +// if !self.info.dinfo.is_null() { +// unsafe { libc::free(self.info.dinfo as *mut _); } +// } +// } +// } + +// Most of this code was adapted from `gstat-rs` (https://github.com/asomers/gstat-rs). +struct GeomSnapshot(NonNull); + +impl GeomSnapshot { + unsafe fn new() -> Option { + match NonNull::new(geom_stats_snapshot_get()) { + Some(n) => Some(Self(n)), + None => { + sysinfo_debug!("geom_stats_snapshot_get failed"); + None + } + } + } + + fn iter(&mut self) -> GeomSnapshotIter { + GeomSnapshotIter(self) + } + + fn reset(&mut self) { + unsafe { geom_stats_snapshot_reset(self.0.as_mut()) } + } +} + +impl Drop for GeomSnapshot { + fn drop(&mut self) { + unsafe { geom_stats_snapshot_free(self.0.as_mut()) }; + } +} + +#[repr(transparent)] +struct Devstat<'a> { + devstat: NonNull, + phantom: PhantomData<&'a devstat>, +} + +struct GeomSnapshotIter<'a>(&'a mut GeomSnapshot); + +impl<'a> Iterator for GeomSnapshotIter<'a> { + type Item = Devstat<'a>; + + fn next(&mut self) -> Option { + let raw = unsafe { geom_stats_snapshot_next(self.0.0.as_mut()) }; + NonNull::new(raw).map(|devstat| Devstat { + devstat, + phantom: PhantomData, + }) + } +} + +impl Drop for GeomSnapshotIter<'_> { + fn drop(&mut self) { + self.0.reset(); + } } diff --git a/src/unix/freebsd/ffi.rs b/src/unix/freebsd/ffi.rs index 1c2aab2e8..305ea8d43 100644 --- a/src/unix/freebsd/ffi.rs +++ b/src/unix/freebsd/ffi.rs @@ -1,15 +1,13 @@ +// Take a look at the license at the top of the repository in the LICENSE file. + #![allow(non_camel_case_types)] -// Because of long double. -#![allow(improper_ctypes)] -use libc::{c_char, c_int, c_long, c_uint, c_void, bintime, kvm_t, CPUSTATES}; +use libc::{c_char, c_int, c_uint, c_void, bintime, kvm_t}; // definitions come from: // https://github.com/freebsd/freebsd-src/blob/main/lib/libdevstat/devstat.h // https://github.com/freebsd/freebsd-src/blob/main/sys/sys/devicestat.h -// 16 bytes says `sizeof` in C so let's use a type of the same size. -pub type c_long_double = u128; pub type devstat_priority = c_int; pub type devstat_support_flags = c_int; pub type devstat_type_flags = c_int; @@ -44,32 +42,43 @@ pub(crate) struct devstat { pub(crate) sequence1: c_uint, } -#[repr(C)] -pub(crate) struct devinfo { - pub(crate) devices: *mut devstat, - pub(crate) mem_ptr: *mut u8, - pub(crate) generation: c_long, - pub(crate) numdevs: c_int, -} +// #[repr(C)] +// pub(crate) struct devinfo { +// pub(crate) devices: *mut devstat, +// pub(crate) mem_ptr: *mut u8, +// pub(crate) generation: c_long, +// pub(crate) numdevs: c_int, +// } -#[repr(C)] -pub(crate) struct statinfo { - pub(crate) cp_time: [c_long; CPUSTATES as usize], - pub(crate) tk_nin: c_long, - pub(crate) tk_nout: c_long, - pub(crate) dinfo: *mut devinfo, - pub(crate) snap_time: c_long_double, -} +// #[repr(C)] +// pub(crate) struct statinfo { +// pub(crate) cp_time: [c_long; CPUSTATES as usize], +// pub(crate) tk_nin: c_long, +// pub(crate) tk_nout: c_long, +// pub(crate) dinfo: *mut devinfo, +// pub(crate) snap_time: c_long_double, +// } pub(crate) const DEVSTAT_N_TRANS_FLAGS: usize = 4; pub(crate) const DEVSTAT_NAME_LEN: usize = 16; +pub(crate) const DEVSTAT_READ: usize = 0x01; +pub(crate) const DEVSTAT_WRITE: usize = 0x02; -pub(crate) const DSM_NONE: c_int = 0; -pub(crate) const DSM_TOTAL_BYTES_READ: c_int = 2; -pub(crate) const DSM_TOTAL_BYTES_WRITE: c_int = 3; +// pub(crate) const DSM_NONE: c_int = 0; +// pub(crate) const DSM_TOTAL_BYTES_READ: c_int = 2; +// pub(crate) const DSM_TOTAL_BYTES_WRITE: c_int = 3; extern "C" { pub(crate) fn devstat_getversion(kd: *mut kvm_t) -> c_int; - pub(crate) fn devstat_getdevs(kd: *mut kvm_t, stats: *mut statinfo) -> c_int; - pub(crate) fn devstat_compute_statistics(current: *mut devstat, previous: *mut devstat, etime: c_long_double, ...) -> c_int; + // pub(crate) fn devstat_getdevs(kd: *mut kvm_t, stats: *mut statinfo) -> c_int; + // pub(crate) fn devstat_compute_statistics(current: *mut devstat, previous: *mut devstat, etime: c_long_double, ...) -> c_int; +} + +#[link(name = "geom")] +extern "C" { + pub(crate) fn geom_stats_open() -> c_int; + pub(crate) fn geom_stats_snapshot_get() -> *mut c_void; + pub(crate) fn geom_stats_snapshot_next(arg: *mut c_void) -> *mut devstat; + pub(crate) fn geom_stats_snapshot_reset(arg: *mut c_void); + pub(crate) fn geom_stats_snapshot_free(arg: *mut c_void); }