|
1 | 1 | extern crate libc;
|
2 | 2 |
|
3 |
| -use std::mem; |
4 | 3 | use std::num::NonZeroUsize;
|
5 | 4 |
|
6 |
| -const PROC_TASKINFO_SIZE: usize = mem::size_of::<libc::proc_taskinfo>(); |
| 5 | +use self::libc::{kern_return_t, mach_msg_type_number_t, mach_port_t, thread_t}; |
| 6 | + |
| 7 | +// This constant is from |
| 8 | +// /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX.sdk/ |
| 9 | +// usr/include/mach/machine/thread_state.h. |
| 10 | +// |
| 11 | +// It has not been updated since Apple devices started to support 64-bit ARM (iOS), so it |
| 12 | +// should be very stable. |
| 13 | +const THREAD_STATE_MAX: i32 = 1296; |
| 14 | +#[allow(non_camel_case_types)] |
| 15 | +// https://github.com/apple/darwin-xnu/blob/a1babec6b135d1f35b2590a1990af3c5c5393479/osfmk/mach/mach_types.defs#L155 |
| 16 | +type task_inspect_t = mach_port_t; |
| 17 | +#[allow(non_camel_case_types)] |
| 18 | +// https://github.com/apple/darwin-xnu/blob/a1babec6b135d1f35b2590a1990af3c5c5393479/osfmk/mach/mach_types.defs#L238 |
| 19 | +type thread_array_t = [thread_t; THREAD_STATE_MAX as usize]; |
| 20 | + |
| 21 | +extern "C" { |
| 22 | + // https://developer.apple.com/documentation/kernel/1537751-task_threads/ |
| 23 | + fn task_threads( |
| 24 | + target_task: task_inspect_t, |
| 25 | + act_list: *mut thread_array_t, |
| 26 | + act_listCnt: *mut mach_msg_type_number_t, |
| 27 | + ) -> kern_return_t; |
| 28 | +} |
7 | 29 |
|
8 | 30 | pub(crate) fn num_threads() -> Option<NonZeroUsize> {
|
9 |
| - let mut pti: libc::proc_taskinfo = unsafe { mem::zeroed() }; |
| 31 | + // http://web.mit.edu/darwin/src/modules/xnu/osfmk/man/task_threads.html |
| 32 | + let mut thread_state = [0u32; THREAD_STATE_MAX as usize]; |
| 33 | + let mut thread_count = 0; |
10 | 34 |
|
11 |
| - let result = unsafe { |
12 |
| - libc::proc_pidinfo( |
13 |
| - libc::getpid(), |
14 |
| - libc::PROC_PIDTASKINFO, |
15 |
| - 0, |
16 |
| - &mut pti as *mut libc::proc_taskinfo as *mut libc::c_void, |
17 |
| - PROC_TASKINFO_SIZE as libc::c_int, |
18 |
| - ) |
19 |
| - }; |
| 35 | + // Safety: `mach_task_self` always returns a valid value, `thread_state` is large enough, and |
| 36 | + // both it and `thread_count` are writable. |
| 37 | + let result = |
| 38 | + unsafe { task_threads(libc::mach_task_self(), &mut thread_state, &mut thread_count) }; |
20 | 39 |
|
21 |
| - if result == PROC_TASKINFO_SIZE as libc::c_int { |
22 |
| - return NonZeroUsize::new(pti.pti_threadnum as usize); |
| 40 | + if result == libc::KERN_SUCCESS { |
| 41 | + NonZeroUsize::new(thread_count as usize) |
| 42 | + } else { |
| 43 | + None |
23 | 44 | }
|
24 |
| - |
25 |
| - None |
26 | 45 | }
|
0 commit comments