Skip to content

Commit

Permalink
Add Main Thread Checker
Browse files Browse the repository at this point in the history
  • Loading branch information
madsmtm committed Dec 7, 2023
1 parent 7fa90c6 commit e06f191
Show file tree
Hide file tree
Showing 8 changed files with 143 additions and 7 deletions.
7 changes: 7 additions & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -13,3 +13,4 @@ directories = "5"
rustc_version = "0.4"
rustc-build-sysroot = "0.4.0"
serde_json = "1.0.87"
apple-sdk = { version = "0.5.1", default-features = false }
8 changes: 7 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -67,7 +67,7 @@ itself.
### Sanitizing

`cargo careful` can additionally build and run your program and standard library
with a sanitizer. This feature is experimental and disabled by default.
with a sanitizer. This feature is experimental and disabled by default.

The [underlying `rustc` feature](https://doc.rust-lang.org/nightly/unstable-book/compiler-flags/sanitizer.html)
doesn't play well with [procedural macros](https://doc.rust-lang.org/reference/procedural-macros.html).
Expand All @@ -86,6 +86,12 @@ setting `ASAN_OPTIONS=detect_leaks=0` in your program's environment, as memory l
usually a soundness or correctness issue. If you set the `ASAN_OPTIONS` environment variable
yourself (to any value, including an empty string), that will override this behavior.

### Main Thread Checker

`cargo careful` automatically enables [Apple's Main Thread Checker](https://developer.apple.com/documentation/xcode/diagnosing-memory-thread-and-crash-issues-early#Detect-improper-UI-updates-on-background-threads) on macOS, iOS, tvOS and watchOS targets.

This helps diagnosing issues with executing thread-unsafe functionality off the main thread on those platforms.

### `cfg` flag

`cargo careful` sets the `careful` configuration flag, so you can use Rust's compile-time
Expand Down
59 changes: 53 additions & 6 deletions src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ use std::ffi::OsString;
use std::path::PathBuf;
use std::process::{self, Command};

use anyhow::{bail, Context, Result};
use anyhow::{anyhow, bail, Context, Result};
use rustc_build_sysroot::{BuildMode, SysrootBuilder, SysrootConfig};
use rustc_version::VersionMeta;

Expand Down Expand Up @@ -38,6 +38,32 @@ pub fn rustc_version_info() -> VersionMeta {
VersionMeta::for_command(rustc()).expect("failed to determine rustc version")
}

/// Find the path for Apple's Main Thread Checker on the current system.
///
/// This is intended to be used on macOS, but should work on other systems
/// that has something similar to XCode set up.
fn main_thread_checker_path() -> Result<Option<PathBuf>> {
if let Some(developer_dir) = apple_sdk::DeveloperDirectory::find_default()
.context("could not find XCode developer directory")?
{
// Introduced in XCode 9.0, and has not changed location since.
// <https://developer.apple.com/library/archive/releasenotes/DeveloperTools/RN-Xcode/Chapters/Introduction.html#//apple_ref/doc/uid/TP40001051-CH1-SW974>
let path = developer_dir
.path()
.join("usr/lib/libMainThreadChecker.dylib");
if path.try_exists()? {
Ok(Some(path))
} else {
Err(anyhow!(
"libMainThreadChecker.dylib could not be found at {}",
path.display()
))
}
} else {
Ok(None)
}
}

// Computes the extra flags that need to be passed to cargo to make it behave like the current
// cargo invocation.
fn cargo_extra_flags() -> Vec<String> {
Expand Down Expand Up @@ -235,7 +261,7 @@ fn build_sysroot(
sysroot_dir
}

fn cargo_careful(args: env::Args) {
fn cargo_careful(args: env::Args) -> Result<()> {
let mut args = args.peekable();

let rustc_version = rustc_version_info();
Expand Down Expand Up @@ -285,6 +311,7 @@ fn cargo_careful(args: env::Args) {
// Forward regular argument.
cargo_args.push(arg);
}

// The rest is for cargo to forward to the binary / test runner.
cargo_args.push("--".into());
cargo_args.extend(args);
Expand Down Expand Up @@ -317,7 +344,7 @@ fn cargo_careful(args: env::Args) {
Some(c) => c,
None => {
// We just did the setup.
return;
return Ok(());
}
};

Expand All @@ -339,6 +366,26 @@ fn cargo_careful(args: env::Args) {
cmd.args(["--target", target.as_str()]);
}

// Enable Main Thread Checker on Apple platforms, as documented here:
// <https://developer.apple.com/documentation/xcode/diagnosing-memory-thread-and-crash-issues-early#Detect-improper-UI-updates-on-background-threads>
let apple_target = target.contains("-darwin")
|| target.contains("-ios")
|| target.contains("-tvos")
|| target.contains("-watchos");
// FIXME: We only do this if the host is running on macOS, since cargo
// doesn't currently have a good way of passing environment variables only
// to target binaries when using `cargo run` or `cargo test` (and this
// means that `rustc` is also run under the main thread checker, which will
// probably fail on other platforms).
// See <https://github.com/rust-lang/cargo/issues/4505>.
if apple_target && cfg!(target_os = "macos") {
if let Some(path) = main_thread_checker_path()? {
cmd.arg("--config");
// TODO: Quote the path correctly according to toml rules
cmd.arg(format!("env.DYLD_INSERT_LIBRARIES={path:?}"));
}
}

cmd.args(cargo_args);

// Setup environment. Both rustc and rustdoc need these flags.
Expand All @@ -357,10 +404,10 @@ fn cargo_careful(args: env::Args) {
}

// Run it!
exec(cmd, (verbose > 0).then_some("[cargo-careful] "));
exec(cmd, (verbose > 0).then_some("[cargo-careful] "))
}

fn main() {
fn main() -> Result<()> {
let mut args = env::args();
// Skip binary name.
args.next().unwrap();
Expand All @@ -373,7 +420,7 @@ fn main() {
match first.as_str() {
"careful" => {
// It's us!
cargo_careful(args);
cargo_careful(args)
}
_ => {
show_error!(
Expand Down
17 changes: 17 additions & 0 deletions test/ci.sh
Original file line number Diff line number Diff line change
Expand Up @@ -18,3 +18,20 @@ cargo careful setup --target x86_64-unknown-none
cargo careful build --target x86_64-unknown-none --locked
cargo clean
popd

# test Apple's Main Thread Checker
if uname -s | grep -q "Darwin"
then
pushd test-main_thread_checker
# Run as normal; this will output warnings, but not fail
cargo careful run --locked
# Run with flag that tells the Main Thread Checker to fail
# See <https://bryce.co/main-thread-checker-configuration/>
if MTC_CRASH_ON_REPORT=1 cargo careful run --locked
then
echo "Main Thread Checker did not crash"
exit 1
fi
cargo clean
popd
fi
32 changes: 32 additions & 0 deletions test/test-main_thread_checker/Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

8 changes: 8 additions & 0 deletions test/test-main_thread_checker/Cargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
[package]
name = "test-cargo-careful-main_thread_checker"
version = "0.1.0"
edition = "2021"
publish = false

[target.'cfg(target_os = "macos")'.dependencies]
objc2 = "0.5.0"
18 changes: 18 additions & 0 deletions test/test-main_thread_checker/src/main.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
//! Run [NSView new] on a separate thread, which should get caught by the
//! main thread checker.
use objc2::rc::Id;
use objc2::runtime::AnyObject;
use objc2::{class, msg_send_id};

#[link(name = "AppKit", kind = "framework")]
extern "C" {}

fn main() {
std::thread::scope(|s| {
s.spawn(|| {
// Note: Usually you'd use `icrate::NSView::new`, this is to
// avoid the heavy dependency.
let _: Id<AnyObject> = unsafe { msg_send_id![class!(NSView), new] };
});
});
}

0 comments on commit e06f191

Please sign in to comment.