// RUN: %clang_cc1 %s -fptrauth-function-pointer-type-discrimination -triple arm64-apple-ios13 -fptrauth-calls -fptrauth-intrinsics -disable-llvm-passes -emit-llvm -o- | FileCheck --check-prefixes=CHECK,TYPE %s // RUN: %clang_cc1 %s -fptrauth-function-pointer-type-discrimination -triple aarch64-linux-gnu -fptrauth-calls -fptrauth-intrinsics -disable-llvm-passes -emit-llvm -o- | FileCheck --check-prefixes=CHECK,TYPE %s // RUN: %clang_cc1 %s -triple arm64-apple-ios13 -fptrauth-calls -fptrauth-intrinsics -disable-llvm-passes -emit-llvm -o- | FileCheck --check-prefixes=CHECK,ZERO %s // RUN: %clang_cc1 %s -triple aarch64-linux-gnu -fptrauth-calls -fptrauth-intrinsics -disable-llvm-passes -emit-llvm -o- | FileCheck --check-prefixes=CHECK,ZERO %s // RUN: %clang_cc1 -xc++ %s -fptrauth-function-pointer-type-discrimination -triple arm64-apple-ios13 -fptrauth-calls -fptrauth-intrinsics -disable-llvm-passes -emit-llvm -o- | FileCheck --check-prefixes=CHECK,TYPE,CHECK-CXX %s // RUN: %clang_cc1 -xc++ %s -fptrauth-function-pointer-type-discrimination -triple aarch64-linux-gnu -fptrauth-calls -fptrauth-intrinsics -disable-llvm-passes -emit-llvm -o- | FileCheck --check-prefixes=CHECK,TYPE,CHECK-CXX %s #ifdef __cplusplus extern "C" { #endif void (*fptr)(void); void (* __ptrauth(0, 0, 42) f2ptr_42_discm)(int); void f(int); void (* const __ptrauth(0, 0, 42) f_const_ptr)(int) = &f; // CHECK-LABEL: define {{.*}}void @test_assign_to_qualified void test_assign_to_qualified() { f2ptr_42_discm = (void (*)(int))fptr; // CHECK: [[ENTRY:.*]]:{{$}} // CHECK: [[FPTR:%.*]] = load ptr, ptr @fptr // CHECK-NEXT: [[CMP:%.*]] = icmp ne ptr [[FPTR]], null // TYPE-NEXT: br i1 [[CMP]], label %[[RESIGN1:.*]], label %[[JOIN1:.*]] // ZERO-NEXT: br i1 [[CMP]], label %[[RESIGN2:.*]], label %[[JOIN2:.*]] // TYPE: [[RESIGN1]]: // TYPE-NEXT: [[FPTR2:%.*]] = ptrtoint ptr [[FPTR]] to i64 // TYPE-NEXT: [[FPTR4:%.*]] = call i64 @llvm.ptrauth.resign(i64 [[FPTR2]], i32 0, i64 18983, i32 0, i64 2712) // TYPE-NEXT: [[FPTR5:%.*]] = inttoptr i64 [[FPTR4]] to ptr // TYPE-NEXT: br label %[[JOIN1]] // TYPE: [[JOIN1]]: // TYPE-NEXT: [[FPTR6:%.*]] = phi ptr [ null, %[[ENTRY]] ], [ [[FPTR5]], %[[RESIGN1]] ] // TYPE-NEXT: [[CMP:%.*]] = icmp ne ptr [[FPTR6]], null // TYPE-NEXT: br i1 [[CMP]], label %[[RESIGN2:.*]], label %[[JOIN2:.*]] // CHECK: [[RESIGN2]]: // TYPE-NEXT: [[FPTR7:%.*]] = ptrtoint ptr [[FPTR6]] to i64 // TYPE-NEXT: [[FPTR8:%.*]] = call i64 @llvm.ptrauth.resign(i64 [[FPTR7]], i32 0, i64 2712, i32 0, i64 42) // ZERO-NEXT: [[FPTR7:%.*]] = ptrtoint ptr [[FPTR]] to i64 // ZERO-NEXT: [[FPTR8:%.*]] = call i64 @llvm.ptrauth.resign(i64 [[FPTR7]], i32 0, i64 0, i32 0, i64 42) // CHECK-NEXT: [[FPTR9:%.*]] = inttoptr i64 [[FPTR8]] to ptr // CHECK-NEXT: br label %[[JOIN2]] // CHECK: [[JOIN2]] // TYPE-NEXT: [[FPTR10:%.*]] = phi ptr [ null, %[[JOIN1]] ], [ [[FPTR9]], %[[RESIGN2]] ] // ZERO-NEXT: [[FPTR10:%.*]] = phi ptr [ null, %[[ENTRY]] ], [ [[FPTR9]], %[[RESIGN2]] ] // CHECK-NEXT store void (i32)* [[FPTR10]], void (i32)** @f2ptr_42_discm } // CHECK-LABEL: define {{.*}}void @test_assign_from_qualified void test_assign_from_qualified() { fptr = (void (*)(void))f2ptr_42_discm; // CHECK: [[ENTRY:.*]]:{{$}} // CHECK: [[FPTR:%.*]] = load ptr, ptr @f2ptr_42_discm // CHECK-NEXT: [[CMP:%.*]] = icmp ne ptr [[FPTR]], null // TYPE-NEXT: br i1 [[CMP]], label %[[RESIGN1:.*]], label %[[JOIN1:.*]] // ZERO-NEXT: br i1 [[CMP]], label %[[RESIGN2:.*]], label %[[JOIN2:.*]] // TYPE: [[RESIGN1]]: // TYPE-NEXT: [[FPTR1:%.*]] = ptrtoint ptr [[FPTR]] to i64 // TYPE-NEXT: [[FPTR2:%.*]] = call i64 @llvm.ptrauth.resign(i64 [[FPTR1]], i32 0, i64 42, i32 0, i64 2712) // TYPE-NEXT: [[FPTR3:%.*]] = inttoptr i64 [[FPTR2]] to ptr // TYPE-NEXT: br label %[[JOIN1]] // TYPE: [[JOIN1]]: // TYPE-NEXT: [[FPTR4:%.*]] = phi ptr [ null, %[[ENTRY]] ], [ [[FPTR3]], %[[RESIGN1]] ] // TYPE-NEXT: [[CMP:%.*]] = icmp ne ptr [[FPTR4]], null // TYPE-NEXT: br i1 [[CMP]], label %[[RESIGN2:.*]], label %[[JOIN2:.*]] // CHECK: [[RESIGN2]]: // TYPE-NEXT: [[FPTR6:%.*]] = ptrtoint ptr [[FPTR4]] to i64 // TYPE-NEXT: [[FPTR7:%.*]] = call i64 @llvm.ptrauth.resign(i64 [[FPTR6]], i32 0, i64 2712, i32 0, i64 18983) // ZERO-NEXT: [[FPTR6:%.*]] = ptrtoint ptr [[FPTR]] to i64 // ZERO-NEXT: [[FPTR7:%.*]] = call i64 @llvm.ptrauth.resign(i64 [[FPTR6]], i32 0, i64 42, i32 0, i64 0) // CHECK-NEXT: [[FPTR8:%.*]] = inttoptr i64 [[FPTR7]] to ptr // CHECK-NEXT: br label %[[JOIN2]] // CHECK: [[JOIN2]] // TYPE-NEXT: [[FPTR9:%.*]] = phi ptr [ null, %[[JOIN1]] ], [ [[FPTR8]], %[[RESIGN2]] ] // ZERO-NEXT: [[FPTR9:%.*]] = phi ptr [ null, %[[ENTRY]] ], [ [[FPTR8]], %[[RESIGN2]] ] // CHECK-NEXT store void ()* [[FPTR10]], void ()** @f2ptr_42_discm } // CHECK-LABEL: define {{.*}}void @test_const_ptr_function_call() void test_const_ptr_function_call(void) { f_const_ptr(1); // TYPE: call void ptrauth (ptr @f, i32 0, i64 2712)(i32 noundef 1) [ "ptrauth"(i32 0, i64 2712) ] // ZERO: call void ptrauth (ptr @f, i32 0)(i32 noundef 1) [ "ptrauth"(i32 0, i64 0) ] } #ifdef __cplusplus void (* get_fptr(void))(int); void (* __ptrauth(0, 0, 42) f_const_ptr2)(int) = get_fptr(); void (* const __ptrauth(0, 1, 43) &f_ref)(int) = f_const_ptr2; // CHECK-CXX-LABEL: define {{.*}}internal void @__cxx_global_var_init() // CHECK-CXX: [[ENTRY:.*]]: // CHECK-CXX: %[[CALL:.*]] = call ptr @get_fptr() // CHECK-CXX: %[[V0:.*]] = icmp ne ptr %[[CALL]], null // CHECK-CXX: br i1 %[[V0]], label %[[RESIGN_NONNULL:.*]], label %[[RESIGN_CONT:.*]] // CHECK-CXX: [[RESIGN_NONNULL]]: // CHECK-CXX: %[[V1:.*]] = ptrtoint ptr %[[CALL]] to i64 // CHECK-CXX: %[[V2:.*]] = call i64 @llvm.ptrauth.resign(i64 %[[V1]], i32 0, i64 2712, i32 0, i64 42) // CHECK-CXX: %[[V3:.*]] = inttoptr i64 %[[V2]] to ptr // CHECK-CXX: br label %[[RESIGN_CONT]] // CHECK-CXX: [[RESIGN_CONT]]: // CHECK-CXX: %[[V4:.*]] = phi ptr [ null, %[[ENTRY]] ], [ %[[V3]], %[[RESIGN_NONNULL]] ] // CHECK-CXX: store ptr %[[V4]], ptr @f_const_ptr2, align 8 // CHECK-CXX-LABEL: define {{.*}}internal void @__cxx_global_var_init.1() // CHECK-CXX: [[ENTRY:.*]]: // CHECK-CXX: %[[V0:.*]] = load ptr, ptr @f_const_ptr2, align 8 // CHECK-CXX: %[[V1:.*]] = call i64 @llvm.ptrauth.blend(i64 ptrtoint (ptr @_ZGR5f_ref_ to i64), i64 43) // CHECK-CXX: %[[V2:.*]] = icmp ne ptr %[[V0]], null // CHECK-CXX: br i1 %[[V2]], label %[[RESIGN_NONNULL:.*]], label %[[RESIGN_CONT:.*]] // CHECK-CXX: [[RESIGN_NONNULL]]: // CHECK-CXX: %[[V3:.*]] = ptrtoint ptr %[[V0]] to i64 // CHECK-CXX: %[[V4:.*]] = call i64 @llvm.ptrauth.resign(i64 %[[V3]], i32 0, i64 42, i32 0, i64 %[[V1]]) // CHECK-CXX: %[[V5:.*]] = inttoptr i64 %[[V4]] to ptr // CHECK-CXX: br label %[[RESIGN_CONT]] // CHECK-CXX: [[RESIGN_CONT]]: // CHECK-CXX: %[[V6:.*]] = phi ptr [ null, %[[ENTRY]] ], [ %[[V5]], %[[RESIGN_NONNULL]] ] // CHECK-CXX: store ptr %[[V6]], ptr @_ZGR5f_ref_, align 8 // CHECK-CXX: store ptr @_ZGR5f_ref_, ptr @f_ref, align 8 // CHECK-CXX-LABEL: define {{.*}}void @test_const_ptr_ref_function_call() void test_const_ptr_ref_function_call(void) { f_ref(1); // CHECK-CXX: %[[V0:.*]] = load ptr, ptr @f_ref, align 8 // CHECK-CXX: %[[V1:.*]] = load ptr, ptr %[[V0]], align 8 // CHECK-CXX: %[[V2:.*]] = ptrtoint ptr %[[V0]] to i64 // CHECK-CXX: %[[V3:.*]] = call i64 @llvm.ptrauth.blend(i64 %[[V2]], i64 43) // CHECK-CXX: call void %[[V1]](i32 noundef 1) [ "ptrauth"(i32 0, i64 %[[V3]]) ] } } #endif