Skip to content

Commit b12a1c8

Browse files
Rewrite MacOS implementation to use public APIs only (#6)
1 parent 9669e08 commit b12a1c8

File tree

2 files changed

+39
-16
lines changed

2 files changed

+39
-16
lines changed

CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,7 @@
1+
# Unreleased
2+
3+
- Modified MacOS implementation to be friendly towards Apple app store review guidelines.
4+
15
# 0.1.2
26

37
- Add support for MacOS

src/macos.rs

+35-16
Original file line numberDiff line numberDiff line change
@@ -1,26 +1,45 @@
11
extern crate libc;
22

3-
use std::mem;
43
use std::num::NonZeroUsize;
54

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+
}
729

830
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;
1034

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) };
2039

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
2344
}
24-
25-
None
2645
}

0 commit comments

Comments
 (0)