//@ compile-flags: -O --target=avr-unknown-gnu-atmega328 --crate-type=rlib -C panic=abort //@ needs-llvm-components: avr // This test validates that function pointers can be stored in global variables // and called upon. It ensures that Rust emits function pointers in the correct // address space to LLVM so that an assertion error relating to casting is // not triggered. // // It also validates that functions can be called through function pointers // through traits. #![feature(no_core, lang_items, intrinsics, unboxed_closures, arbitrary_self_types)] #![crate_type = "lib"] #![no_core] #[lang = "sized"] pub trait Sized {} #[lang = "copy"] pub trait Copy {} impl Copy for *const T {} #[lang = "receiver"] pub trait Receiver {} #[lang = "tuple_trait"] pub trait Tuple {} pub struct Result { _a: T, _b: E, } impl Copy for usize {} impl Copy for &usize {} #[lang = "drop_in_place"] pub unsafe fn drop_in_place(_: *mut T) {} #[lang = "fn_once"] pub trait FnOnce { #[lang = "fn_once_output"] type Output; extern "rust-call" fn call_once(self, args: Args) -> Self::Output; } #[lang = "fn_mut"] pub trait FnMut: FnOnce { extern "rust-call" fn call_mut(&mut self, args: Args) -> Self::Output; } #[lang = "fn"] pub trait Fn: FnOnce { /// Performs the call operation. extern "rust-call" fn call(&self, args: Args) -> Self::Output; } extern "rust-intrinsic" { pub fn transmute(src: Src) -> Dst; } pub static mut STORAGE_FOO: fn(&usize, &mut u32) -> Result<(), ()> = arbitrary_black_box; pub static mut STORAGE_BAR: u32 = 12; fn arbitrary_black_box(ptr: &usize, _: &mut u32) -> Result<(), ()> { let raw_ptr = ptr as *const usize; let _v: usize = unsafe { *raw_ptr }; loop {} } #[inline(never)] #[no_mangle] fn call_through_fn_trait(a: &mut impl Fn<(), Output = ()>) { (*a)() } #[inline(never)] fn update_bar_value() { unsafe { STORAGE_BAR = 88; } } // CHECK: define dso_local void @test(){{.+}}addrspace(1) #[no_mangle] pub extern "C" fn test() { let mut buf = 7; // A call through the Fn trait must use address space 1. // // CHECK: call{{.+}}addrspace(1) void @call_through_fn_trait() call_through_fn_trait(&mut update_bar_value); // A call through a global variable must use address space 1. // CHECK: load {{.*}}addrspace(1){{.+}}FOO unsafe { STORAGE_FOO(&1, &mut buf); } } // Validate that we can codegen transmutes between data ptrs and fn ptrs. // CHECK: define{{.+}}ptr addrspace(1) @transmute_data_ptr_to_fn(ptr{{.*}} %x) #[no_mangle] pub unsafe fn transmute_data_ptr_to_fn(x: *const ()) -> fn() { // It doesn't matter precisely how this is codegenned (through memory or an addrspacecast), // as long as it doesn't cause a verifier error by using `bitcast`. transmute(x) } // CHECK: define{{.+}}ptr @transmute_fn_ptr_to_data(ptr addrspace(1){{.*}} %x) #[no_mangle] pub unsafe fn transmute_fn_ptr_to_data(x: fn()) -> *const () { // It doesn't matter precisely how this is codegenned (through memory or an addrspacecast), // as long as it doesn't cause a verifier error by using `bitcast`. transmute(x) } pub enum Either { A(T), B(U), } // Previously, we would codegen this as passing/returning a scalar pair of `{ i8, ptr }`, // with the `ptr` field representing both `&i32` and `fn()` depending on the variant. // This is incorrect, because `fn()` should be `ptr addrspace(1)`, not `ptr`. // CHECK: define{{.+}}void @should_not_combine_addrspace(ptr{{.+}}sret{{.+}}%_0, ptr{{.+}}%x) #[no_mangle] #[inline(never)] pub fn should_not_combine_addrspace(x: Either<&i32, fn()>) -> Either<&i32, fn()> { x } // The incorrectness described above would result in us producing (after optimizations) // a `ptrtoint`/`inttoptr` roundtrip to convert from `ptr` to `ptr addrspace(1)`. // CHECK-LABEL: @call_with_fn_ptr #[no_mangle] pub fn call_with_fn_ptr<'a>(f: fn()) -> Either<&'a i32, fn()> { // CHECK-NOT: ptrtoint // CHECK-NOT: inttoptr // CHECK: call addrspace(1) void @should_not_combine_addrspace should_not_combine_addrspace(Either::B(f)) }