2024-01-20 18:48:41 -08:00
|
|
|
use std::cell::{Cell, RefCell};
|
2016-06-19 15:58:40 +02:00
|
|
|
use std::sync::{Arc, Mutex};
|
2020-11-22 09:08:04 +01:00
|
|
|
use std::thread;
|
|
|
|
use std::time::Duration;
|
2016-06-19 15:58:40 +02:00
|
|
|
|
2020-11-22 09:08:04 +01:00
|
|
|
#[test]
|
2024-09-04 02:18:32 +02:00
|
|
|
#[cfg_attr(any(target_os = "emscripten", target_os = "wasi"), ignore)] // no threads
|
2024-04-05 23:22:17 +02:00
|
|
|
#[cfg_attr(miri, ignore)] // Miri does not like the thread leak
|
|
|
|
fn sleep_very_long() {
|
2016-06-19 15:58:40 +02:00
|
|
|
let finished = Arc::new(Mutex::new(false));
|
|
|
|
let t_finished = finished.clone();
|
|
|
|
thread::spawn(move || {
|
2020-11-22 09:08:04 +01:00
|
|
|
thread::sleep(Duration::new(u64::MAX, 0));
|
2016-06-19 15:58:40 +02:00
|
|
|
*t_finished.lock().unwrap() = true;
|
|
|
|
});
|
2020-11-22 09:08:04 +01:00
|
|
|
thread::sleep(Duration::from_millis(100));
|
2016-06-19 15:58:40 +02:00
|
|
|
assert_eq!(*finished.lock().unwrap(), false);
|
|
|
|
}
|
2024-01-20 18:48:41 -08:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn thread_local_containing_const_statements() {
|
|
|
|
// This exercises the `const $init:block` cases of the thread_local macro.
|
|
|
|
// Despite overlapping with expression syntax, the `const { ... }` is not
|
|
|
|
// parsed as `$init:expr`.
|
|
|
|
thread_local! {
|
|
|
|
static CELL: Cell<u32> = const {
|
|
|
|
let value = 1;
|
|
|
|
Cell::new(value)
|
|
|
|
};
|
|
|
|
|
|
|
|
static REFCELL: RefCell<u32> = const {
|
|
|
|
let value = 1;
|
|
|
|
RefCell::new(value)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
assert_eq!(CELL.get(), 1);
|
|
|
|
assert_eq!(REFCELL.take(), 1);
|
|
|
|
}
|
2024-09-22 22:15:21 +01:00
|
|
|
|
2024-10-18 10:12:11 +08:00
|
|
|
#[test]
|
|
|
|
fn thread_local_hygeiene() {
|
|
|
|
// Previously `thread_local_inner!` had use imports for `LocalKey`, `Storage`, `EagerStorage`
|
|
|
|
// and `LazyStorage`. The use imports will shadow a user-provided type or type alias if the
|
|
|
|
// user-provided type or type alias has the same name. Make sure that this does not happen. See
|
|
|
|
// <https://github.com/rust-lang/rust/issues/131863>.
|
|
|
|
//
|
|
|
|
// NOTE: if the internal implementation details change (i.e. get renamed), this test should be
|
|
|
|
// updated.
|
|
|
|
|
|
|
|
#![allow(dead_code)]
|
|
|
|
type LocalKey = ();
|
|
|
|
type Storage = ();
|
|
|
|
type LazyStorage = ();
|
|
|
|
type EagerStorage = ();
|
|
|
|
thread_local! {
|
|
|
|
static A: LocalKey = const { () };
|
|
|
|
static B: Storage = const { () };
|
|
|
|
static C: LazyStorage = const { () };
|
|
|
|
static D: EagerStorage = const { () };
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-22 22:15:21 +01:00
|
|
|
#[test]
|
|
|
|
// Include an ignore list on purpose, so that new platforms don't miss it
|
|
|
|
#[cfg_attr(
|
|
|
|
any(
|
|
|
|
target_os = "redox",
|
|
|
|
target_os = "l4re",
|
|
|
|
target_env = "sgx",
|
|
|
|
target_os = "solid_asp3",
|
|
|
|
target_os = "teeos",
|
|
|
|
target_os = "wasi"
|
|
|
|
),
|
|
|
|
should_panic
|
|
|
|
)]
|
|
|
|
fn available_parallelism() {
|
|
|
|
// check that std::thread::available_parallelism() returns a valid value
|
|
|
|
assert!(thread::available_parallelism().is_ok());
|
|
|
|
}
|