1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58
|
//
// Print the physical memory page with the most references
//
// Require CAP_SYS_ADMIN privilege, or root
//
// Sample output:
//
// Found RAM here: 0x1000-0x9fbff
// Lots of references to this locations: addr=0x9d000, pfn=157, refs=0
// Found RAM here: 0x100000-0xdffeffff
// Lots of references to this locations: addr=0x81ba3000, pfn=531363, refs=128
// Found RAM here: 0x100000000-0x11fffffff
// Lots of references to this locations: addr=0x1b575000, pfn=111989, refs=134
//
use procfs::prelude::*;
fn main() {
if !rustix::process::geteuid().is_root() {
panic!("ERROR: Access to /proc/iomem requires root, re-run with sudo");
}
let page_size = procfs::page_size();
// /proc/iomem contain a list of memory mapping, but we're only interested in RAM mapping
let iomem = procfs::iomem().expect("Can't open /proc/iomem");
let ram = iomem
.iter()
.filter_map(|(_, map)| if map.name == "System RAM" { Some(map) } else { None });
let mut kpagecount = procfs::KPageCount::new().expect("Can't open /proc/kpagecount");
for map in ram {
println!("Found RAM here: 0x{:x}-0x{:x}", map.address.0, map.address.1);
// Physical memory is divided into pages of `page_size` bytes (usually 4kiB)
// Each page is referenced by its Page Fram Number (PFN)
let (start_pfn, end_pfn) = map.get_range().get();
let page_references = kpagecount
.get_count_in_range(start_pfn, end_pfn)
.expect("Can't read from /proc/kpagecount");
// find the page with most references
let (pfn, refs) = page_references
.iter()
.enumerate()
.max_by(|(_, a), (_, b)| a.cmp(b))
.unwrap();
println!(
"Lots of references to this locations: addr=0x{:x}, pfn={}, refs={}",
pfn * page_size as usize,
pfn,
refs
);
}
}
|