85 lines
2.0 KiB
Rust
85 lines
2.0 KiB
Rust
//@ only-aarch64
|
|
//@ only-linux
|
|
//@ needs-asm-support
|
|
//@ run-pass
|
|
|
|
#![feature(thread_local)]
|
|
|
|
use std::arch::asm;
|
|
|
|
extern "C" fn f1() -> i32 {
|
|
111
|
|
}
|
|
|
|
// The compiler will generate a shim to hide the caller location parameter.
|
|
#[track_caller]
|
|
fn f2() -> i32 {
|
|
222
|
|
}
|
|
|
|
macro_rules! call {
|
|
($func:path) => {
|
|
unsafe {
|
|
let result: i32;
|
|
asm!("bl {}", sym $func,
|
|
out("w0") result,
|
|
out("x20") _, out("x21") _, out("x22") _,
|
|
out("x23") _, out("x24") _, out("x25") _,
|
|
out("x26") _, out("x27") _, out("x28") _,
|
|
);
|
|
result
|
|
}
|
|
}
|
|
}
|
|
|
|
macro_rules! static_addr {
|
|
($s:expr) => {
|
|
unsafe {
|
|
let result: *const u32;
|
|
asm!(
|
|
// ADRP gives the address of a 4KB page from a PC-relative address
|
|
"adrp {out}, {sym}",
|
|
// We then add the remaining lower 12 bits
|
|
"add {out}, {out}, #:lo12:{sym}",
|
|
out = out(reg) result,
|
|
sym = sym $s);
|
|
result
|
|
}
|
|
}
|
|
}
|
|
macro_rules! static_tls_addr {
|
|
($s:expr) => {
|
|
unsafe {
|
|
let result: *const u32;
|
|
asm!(
|
|
// Load the thread pointer register
|
|
"mrs {out}, TPIDR_EL0",
|
|
// Add the top 12 bits of the symbol's offset
|
|
"add {out}, {out}, :tprel_hi12:{sym}",
|
|
// And the bottom 12 bits
|
|
"add {out}, {out}, :tprel_lo12_nc:{sym}",
|
|
out = out(reg) result,
|
|
sym = sym $s
|
|
);
|
|
result
|
|
}
|
|
}
|
|
}
|
|
|
|
static S1: u32 = 111;
|
|
#[thread_local]
|
|
static S2: u32 = 222;
|
|
|
|
fn main() {
|
|
assert_eq!(call!(f1), 111);
|
|
assert_eq!(call!(f2), 222);
|
|
assert_eq!(static_addr!(S1), &S1 as *const u32);
|
|
assert_eq!(static_tls_addr!(S2), &S2 as *const u32);
|
|
std::thread::spawn(|| {
|
|
assert_eq!(static_addr!(S1), &S1 as *const u32);
|
|
assert_eq!(static_tls_addr!(S2), &S2 as *const u32);
|
|
})
|
|
.join()
|
|
.unwrap();
|
|
}
|