diff options
author | bors[bot] <26634292+bors[bot]@users.noreply.github.com> | 2021-06-11 23:00:23 +0100 |
---|---|---|
committer | GitHub <[email protected]> | 2021-06-11 23:00:23 +0100 |
commit | 409f5fb563bfc992e4d05245d5c17e77ff8e2762 (patch) | |
tree | 934302b3a697fdf8c3b27e4885e75fb5b649f5d5 /crates/profile/src | |
parent | 72ea02869b94caf6dd450e17c4998357849ab3c5 (diff) | |
parent | 74dc9bb0e9c1d62b3faa33b204abb04a17e57bcf (diff) |
Merge #9204
9204: feat: more accurate memory usage info on glibc Linux r=jonas-schievink a=jonas-schievink
This adds support for the new `mallinfo2` API added in glibc 2.33. It addresses a shortcoming in the `mallinfo` API where it was unable to handle memory usage of more than 2 GB, which we sometimes exceed.
Blocked on https://github.com/rust-lang/libc/pull/2228
Co-authored-by: Jonas Schievink <[email protected]>
Diffstat (limited to 'crates/profile/src')
-rw-r--r-- | crates/profile/src/memory_usage.rs | 35 |
1 files changed, 32 insertions, 3 deletions
diff --git a/crates/profile/src/memory_usage.rs b/crates/profile/src/memory_usage.rs index 6ef58c9c1..0b0a196ae 100644 --- a/crates/profile/src/memory_usage.rs +++ b/crates/profile/src/memory_usage.rs | |||
@@ -32,9 +32,7 @@ impl MemoryUsage { | |||
32 | allocated: Bytes(jemalloc_ctl::stats::allocated::read().unwrap() as isize), | 32 | allocated: Bytes(jemalloc_ctl::stats::allocated::read().unwrap() as isize), |
33 | } | 33 | } |
34 | } else if #[cfg(all(target_os = "linux", target_env = "gnu"))] { | 34 | } else if #[cfg(all(target_os = "linux", target_env = "gnu"))] { |
35 | // Note: This is incredibly slow. | 35 | memusage_linux() |
36 | let alloc = unsafe { libc::mallinfo() }.uordblks as isize; | ||
37 | MemoryUsage { allocated: Bytes(alloc) } | ||
38 | } else if #[cfg(windows)] { | 36 | } else if #[cfg(windows)] { |
39 | // There doesn't seem to be an API for determining heap usage, so we try to | 37 | // There doesn't seem to be an API for determining heap usage, so we try to |
40 | // approximate that by using the Commit Charge value. | 38 | // approximate that by using the Commit Charge value. |
@@ -58,6 +56,37 @@ impl MemoryUsage { | |||
58 | } | 56 | } |
59 | } | 57 | } |
60 | 58 | ||
59 | #[cfg(all(target_os = "linux", target_env = "gnu"))] | ||
60 | fn memusage_linux() -> MemoryUsage { | ||
61 | // Linux/glibc has 2 APIs for allocator introspection that we can use: mallinfo and mallinfo2. | ||
62 | // mallinfo uses `int` fields and cannot handle memory usage exceeding 2 GB. | ||
63 | // mallinfo2 is very recent, so its presence needs to be detected at runtime. | ||
64 | // Both are abysmally slow. | ||
65 | |||
66 | use std::ffi::CStr; | ||
67 | use std::sync::atomic::{AtomicUsize, Ordering}; | ||
68 | |||
69 | static MALLINFO2: AtomicUsize = AtomicUsize::new(1); | ||
70 | |||
71 | let mut mallinfo2 = MALLINFO2.load(Ordering::Relaxed); | ||
72 | if mallinfo2 == 1 { | ||
73 | let cstr = CStr::from_bytes_with_nul(b"mallinfo2\0").unwrap(); | ||
74 | mallinfo2 = unsafe { libc::dlsym(libc::RTLD_DEFAULT, cstr.as_ptr()) } as usize; | ||
75 | // NB: races don't matter here, since they'll always store the same value | ||
76 | MALLINFO2.store(mallinfo2, Ordering::Relaxed); | ||
77 | } | ||
78 | |||
79 | if mallinfo2 == 0 { | ||
80 | // mallinfo2 does not exist, use mallinfo. | ||
81 | let alloc = unsafe { libc::mallinfo() }.uordblks as isize; | ||
82 | MemoryUsage { allocated: Bytes(alloc) } | ||
83 | } else { | ||
84 | let mallinfo2: fn() -> libc::mallinfo2 = unsafe { std::mem::transmute(mallinfo2) }; | ||
85 | let alloc = mallinfo2().uordblks as isize; | ||
86 | MemoryUsage { allocated: Bytes(alloc) } | ||
87 | } | ||
88 | } | ||
89 | |||
61 | #[derive(Default, PartialEq, Eq, PartialOrd, Ord, Hash, Clone, Copy)] | 90 | #[derive(Default, PartialEq, Eq, PartialOrd, Ord, Hash, Clone, Copy)] |
62 | pub struct Bytes(isize); | 91 | pub struct Bytes(isize); |
63 | 92 | ||