// RUN: %clang_cc1 %s -triple arm64e-apple-ios13 -fptrauth-calls -fptrauth-intrinsics -emit-llvm -o- -fptrauth-function-pointer-type-discrimination | FileCheck -check-prefixes CHECK,TYPE %s // RUN: %clang_cc1 %s -triple aarch64-linux-gnu -fptrauth-calls -fptrauth-intrinsics -emit-llvm -o- -fptrauth-function-pointer-type-discrimination | FileCheck -check-prefixes CHECK,TYPE %s // RUN: %clang_cc1 %s -triple arm64e-apple-ios13 -fptrauth-calls -fptrauth-intrinsics -emit-llvm -o- | FileCheck -check-prefixes CHECK,ZERO %s // RUN: %clang_cc1 %s -triple aarch64-linux-gnu -fptrauth-calls -fptrauth-intrinsics -emit-llvm -o- | FileCheck -check-prefixes CHECK,ZERO %s typedef void (*fptr_t)(void); char *cptr; void (*fptr)(void); // CHECK-LABEL: define{{.*}} void @test1 void test1() { // TYPE: [[LOAD:%.*]] = load ptr, ptr @cptr // TYPE: [[TOINT:%.*]] = ptrtoint ptr [[LOAD]] to i64 // TYPE: call i64 @llvm.ptrauth.resign(i64 [[TOINT]], i32 0, i64 0, i32 0, i64 18983) // TYPE: call void {{.*}}() [ "ptrauth"(i32 0, i64 18983) ] // ZERO-NOT: @llvm.ptrauth.resign (*(fptr_t)cptr)(); } // CHECK-LABEL: define{{.*}} i8 @test2 char test2() { return *(char *)fptr; // TYPE: [[LOAD:%.*]] = load ptr, ptr @fptr // TYPE: [[CMP:%.*]] = icmp ne ptr [[LOAD]], null // TYPE-NEXT: br i1 [[CMP]], label %[[NONNULL:.*]], label %[[CONT:.*]] // TYPE: [[NONNULL]]: // TYPE: [[TOINT:%.*]] = ptrtoint ptr [[LOAD]] to i64 // TYPE: [[CALL:%.*]] = call i64 @llvm.ptrauth.resign(i64 [[TOINT]], i32 0, i64 18983, i32 0, i64 0) // TYPE: [[TOPTR:%.*]] = inttoptr i64 [[CALL]] to ptr // TYPE: [[CONT]]: // TYPE: phi ptr [ null, {{.*}} ], [ [[TOPTR]], %[[NONNULL]] ] // ZERO-NOT: @llvm.ptrauth.resign } // CHECK-LABEL: define{{.*}} void @test4 void test4() { (*((fptr_t)(&*((char *)(&*(fptr_t)cptr)))))(); // CHECK: [[LOAD:%.*]] = load ptr, ptr @cptr // TYPE-NEXT: [[CAST4:%.*]] = ptrtoint ptr [[LOAD]] to i64 // TYPE-NEXT: [[RESIGN:%.*]] = call i64 @llvm.ptrauth.resign(i64 [[CAST4]], i32 0, i64 0, i32 0, i64 18983) // TYPE-NEXT: [[CAST5:%.*]] = inttoptr i64 [[RESIGN]] to ptr // TYPE-NEXT: call void [[CAST5]]() [ "ptrauth"(i32 0, i64 18983) ] // ZERO-NOT: @llvm.ptrauth.resign // ZERO: call void [[LOAD]]() [ "ptrauth"(i32 0, i64 0) ] } void *vptr; // CHECK-LABEL: define{{.*}} void @test5 void test5() { vptr = &*(char *)fptr; // TYPE: [[LOAD:%.*]] = load ptr, ptr @fptr // TYPE-NEXT: [[CMP]] = icmp ne ptr [[LOAD]], null // TYPE-NEXT: br i1 [[CMP]], label %[[NONNULL:.*]], label %[[CONT:.*]] // TYPE: [[NONNULL]]: // TYPE: [[RESIGN:%.*]] = call i64 @llvm.ptrauth.resign(i64 {{.*}}, i32 0, i64 18983, i32 0, i64 0) // TYPE: [[CAST:%.*]] = inttoptr i64 [[RESIGN]] to ptr // TYPE: [[CONT]]: // TYPE: [[PHI:%.*]] = phi ptr [ null, {{.*}} ], [ [[CAST]], %[[NONNULL]] ] // TYPE: store ptr [[PHI]], ptr @vptr // ZERO-NOT: @llvm.ptrauth.resign }