// RUN: %clang_cc1 -triple arm64-apple-ios -fptrauth-calls -fptrauth-intrinsics -emit-llvm %s -O0 -o - | FileCheck %s #define __ptrauth(...) __ptrauth(__VA_ARGS__) __INTPTR_TYPE__ __ptrauth(1, 0, 56) g1 = 0; // CHECK: @g1 = global i64 0 __INTPTR_TYPE__ __ptrauth(1, 1, 1272) g2 = 0; // CHECK: @g2 = global i64 0 extern __UINTPTR_TYPE__ test_int; __UINTPTR_TYPE__ __ptrauth(3, 1, 23) g3 = (__UINTPTR_TYPE__)&test_int; // CHECK: @test_int = external global i64 // CHECK: @g3 = global i64 ptrtoint (ptr ptrauth (ptr @test_int, i32 3, i64 23, ptr @g3) to i64) __INTPTR_TYPE__ __ptrauth(1, 1, 712) ga[3] = {0,0,(__UINTPTR_TYPE__)&test_int}; // CHECK: @ga = global [3 x i64] [i64 0, i64 0, i64 ptrtoint (ptr ptrauth (ptr @test_int, i32 1, i64 712, ptr getelementptr inbounds ([3 x i64], ptr @ga, i32 0, i32 2)) to i64)] struct A { __INTPTR_TYPE__ __ptrauth(1, 0, 431) f0; __INTPTR_TYPE__ __ptrauth(1, 0, 9182) f1; __INTPTR_TYPE__ __ptrauth(1, 0, 783) f2; }; struct A gs1 = {0, 0, (__UINTPTR_TYPE__)&test_int}; // CHECK: @gs1 = global %struct.A { i64 0, i64 0, i64 ptrtoint (ptr ptrauth (ptr @test_int, i32 1, i64 783) to i64) } struct B { __INTPTR_TYPE__ __ptrauth(1, 1, 1276) f0; __INTPTR_TYPE__ __ptrauth(1, 1, 23674) f1; __INTPTR_TYPE__ __ptrauth(1, 1, 163) f2; }; struct B gs2 = {0, 0, (__UINTPTR_TYPE__)&test_int}; // CHECK: @gs2 = global %struct.B { i64 0, i64 0, i64 ptrtoint (ptr ptrauth (ptr @test_int, i32 1, i64 163, ptr getelementptr inbounds (%struct.B, ptr @gs2, i32 0, i32 2)) to i64) } // CHECK-LABEL: i64 @test_read_globals __INTPTR_TYPE__ test_read_globals() { __INTPTR_TYPE__ result = g1 + g2 + g3; // CHECK: [[A:%.*]] = load i64, ptr @g1 // CHECK: [[AUTHED:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[A]], i32 1, i64 56) // CHECK: [[B:%.*]] = load i64, ptr @g2 // CHECK: [[BLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 ptrtoint (ptr @g2 to i64), i64 1272) // CHECK: [[AUTHED:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[B]], i32 1, i64 [[BLENDED]]) // CHECK: [[VALUE:%.*]] = load i64, ptr @g3 // CHECK: [[BLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 ptrtoint (ptr @g3 to i64), i64 23) // CHECK: [[AUTHED:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[VALUE]], i32 3, i64 [[BLENDED]]) for (int i = 0; i < 3; i++) { result += ga[i]; } // CHECK: for.cond: // CHECK: [[TEMP:%.*]] = load i32, ptr [[IDX_ADDR:%.*]] // CHECK: for.body: // CHECK: [[IDX:%.*]] = load i32, ptr [[IDX_ADDR]] // CHECK: [[IDXPROM:%.*]] = sext i32 [[IDX]] to i64 // CHECK: [[ARRAYIDX:%.*]] = getelementptr inbounds [3 x i64], ptr @ga, i64 0, i64 [[IDXPROM]] // CHECK: [[VALUE:%.*]] = load i64, ptr [[ARRAYIDX]] // CHECK: [[CASTIDX:%.*]] = ptrtoint ptr [[ARRAYIDX]] to i64 // CHECK: [[BLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 [[CASTIDX]], i64 712) // CHECK: resign.nonnull6: // CHECK: [[AUTHED:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[VALUE]], i32 1, i64 [[BLENDED]]) // CHECK: resign.cont7 result += gs1.f0 + gs1.f1 + gs1.f2; // CHECK: resign.cont10: // CHECK: [[ADDR:%.*]] = load i64, ptr getelementptr inbounds nuw (%struct.A, ptr @gs1, i32 0, i32 1 // CHECK: resign.nonnull11: // CHECK: [[AUTHED:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[ADDR]], i32 1, i64 9182) // CHECK: resign.cont12: // CHECK: [[ADDR:%.*]] = load i64, ptr getelementptr inbounds nuw (%struct.A, ptr @gs1, i32 0, i32 2) // CHECK: [[AUTHED:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[ADDR]], i32 1, i64 783) result += gs2.f0 + gs2.f1 + gs2.f2; // CHECK: [[ADDR:%.*]] = load i64, ptr @gs2 // CHECK: [[BLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 ptrtoint (ptr @gs2 to i64), i64 1276) // CHECK: [[AUTHED:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[ADDR]], i32 1, i64 [[BLENDED]]) // CHECK: [[ADDR:%.*]] = load i64, ptr getelementptr inbounds nuw (%struct.B, ptr @gs2, i32 0, i32 1) // CHECK: [[BLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 ptrtoint (ptr getelementptr inbounds nuw (%struct.B, ptr @gs2, i32 0, i32 1) to i64), i64 23674) // CHECK: [[AUTHED:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[ADDR]], i32 1, i64 [[BLENDED]]) // CHECK: [[ADDR:%.*]] = load i64, ptr getelementptr inbounds nuw (%struct.B, ptr @gs2, i32 0, i32 2) // CHECK: [[BLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 ptrtoint (ptr getelementptr inbounds nuw (%struct.B, ptr @gs2, i32 0, i32 2) to i64), i64 163) return result; } // CHECK-LABEL: void @test_write_globals void test_write_globals(int i, __INTPTR_TYPE__ j) { g1 = i; g2 = j; g3 = 0; ga[0] = i; ga[1] = j; ga[2] = 0; gs1.f0 = i; gs1.f1 = j; gs1.f2 = 0; gs2.f0 = i; gs2.f1 = j; gs2.f2 = 0; } // CHECK-LABEL: define void @test_set_A void test_set_A(struct A *a, __INTPTR_TYPE__ x, int y) { a->f0 = x; // CHECK: [[XADDR:%.*]] = load i64, ptr %x.addr // CHECK: [[SIGNED_X:%.*]] = call i64 @llvm.ptrauth.sign(i64 [[XADDR]], i32 1, i64 431) a->f1 = y; // CHECK: [[Y:%.*]] = load i32, ptr %y.addr // CHECK: [[CONV:%.*]] = sext i32 [[Y]] to i64 // CHECK: [[SIGNED:%.*]] = call i64 @llvm.ptrauth.sign(i64 [[CONV]], i32 1, i64 9182) a->f2 = 0; // CHECK: [[A:%.*]] = load ptr, ptr %a.addr // CHECK: [[F2:%.*]] = getelementptr inbounds nuw %struct.A, ptr [[A]], i32 0, i32 2 // CHECK: store i64 0, ptr [[F2]] } // CHECK-LABEL: define void @test_set_B void test_set_B(struct B *b, __INTPTR_TYPE__ x, int y) { b->f0 = x; // CHECK: [[X:%.*]] = load i64, ptr %x.addr // CHECK: [[F0_ADDR:%.*]] = ptrtoint ptr %f0 to i64 // CHECK: [[BLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 [[F0_ADDR]], i64 1276) // CHECK: [[SIGNED:%.*]] = call i64 @llvm.ptrauth.sign(i64 [[X]], i32 1, i64 [[BLENDED]]) b->f1 = y; // CHECK: [[B:%.*]] = load ptr, ptr %b.addr // CHECK: [[F1_ADDR:%.*]] = getelementptr inbounds nuw %struct.B, ptr [[B]], i32 0, i32 1 // CHECK: [[Y:%.*]] = load i32, ptr %y.addr, align 4 // CHECK: [[CONV:%.*]] = sext i32 [[Y]] to i64 // CHECK: [[CAST_ADDR:%.*]] = ptrtoint ptr [[F1_ADDR]] to i64 // CHECK: [[BLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 [[CAST_ADDR]], i64 23674) // CHECK: [[SIGNED:%.*]] = call i64 @llvm.ptrauth.sign(i64 [[CONV]], i32 1, i64 [[BLENDED]]) b->f2 = 0; // CHECK: [[B:%.*]] = load ptr, ptr %b.addr // CHECK: [[F2_ADDR:%.*]] = getelementptr inbounds nuw %struct.B, ptr [[B]], i32 0, i32 2 // CHECK: store i64 0, ptr [[F2_ADDR]] } // CHECK-LABEL: define i64 @test_get_A __INTPTR_TYPE__ test_get_A(struct A *a) { return a->f0 + a->f1 + a->f2; // CHECK: [[A:%.*]] = load ptr, ptr %a.addr // CHECK: [[F0_ADDR:%.*]] = getelementptr inbounds nuw %struct.A, ptr [[A]], i32 0, i32 0 // CHECK: [[F0:%.*]] = load i64, ptr [[F0_ADDR]] // CHECK: [[AUTH:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[F0]], i32 1, i64 431) // CHECK: [[A:%.*]] = load ptr, ptr %a.addr // CHECK: [[F1_ADDR:%.*]] = getelementptr inbounds nuw %struct.A, ptr [[A]], i32 0, i32 1 // CHECK: [[F1:%.*]] = load i64, ptr [[F1_ADDR]] // CHECK: [[AUTH:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[F1]], i32 1, i64 9182) // CHECK: [[A:%.*]] = load ptr, ptr %a.addr // CHECK: [[F2_ADDR:%.*]] = getelementptr inbounds nuw %struct.A, ptr [[A]], i32 0, i32 2 // CHECK: [[F2:%.*]] = load i64, ptr [[F2_ADDR]] // CHECK: [[AUTH:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[F2]], i32 1, i64 783) } // CHECK-LABEL: define i64 @test_get_B __INTPTR_TYPE__ test_get_B(struct B *b) { return b->f0 + b->f1 + b->f2; // CHECK: [[B:%.*]] = load ptr, ptr %b.addr // CHECK: [[F0:%.*]] = getelementptr inbounds nuw %struct.B, ptr [[B]], i32 0, i32 0 // CHECK: [[VALUE:%.*]] = load i64, ptr [[F0]] // CHECK: [[CASTF0:%.*]] = ptrtoint ptr %f0 to i64 // CHECK: [[BLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 [[CASTF0]], i64 1276) // CHECK: [[AUTHED:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[VALUE]], i32 1, i64 [[BLENDED]]) // CHECK: [[B:%.*]] = load ptr, ptr %b.addr // CHECK: [[ADDR:%.*]] = getelementptr inbounds nuw %struct.B, ptr [[B]], i32 0, i32 1 // CHECK: [[VALUE:%.*]] = load i64, ptr [[ADDR]] // CHECK: [[CAST_ADDR:%.*]] = ptrtoint ptr [[ADDR]] to i64 // CHECK: [[BLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 [[CAST_ADDR]], i64 23674) // CHECK: [[AUTHED:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[VALUE]], i32 1, i64 [[BLENDED]]) // CHECK: [[B:%.*]] = load ptr, ptr %b.addr // CHECK: [[ADDR:%.*]] = getelementptr inbounds nuw %struct.B, ptr [[B]], i32 0, i32 2 // CHECK: [[VALUE:%.*]] = load i64, ptr [[ADDR]] // CHECK: [[CAST_ADDR:%.*]] = ptrtoint ptr [[ADDR]] to i64 // CHECK: [[BLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 [[CAST_ADDR]], i64 163) // CHECK: [[AUTHED:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[VALUE]], i32 1, i64 [[BLENDED]]) } // CHECK-LABEL: define void @test_resign void test_resign(struct A* a, const struct B *b) { a->f0 = b->f0; // CHECK: [[A:%.*]] = load ptr, ptr %a.addr, align 8 // CHECK: [[F0:%.*]] = getelementptr inbounds nuw %struct.A, ptr [[A]], i32 0, i32 0 // CHECK: [[B:%.*]] = load ptr, ptr %b.addr, align 8 // CHECK: [[F01:%.*]] = getelementptr inbounds nuw %struct.B, ptr [[B]], i32 0, i32 0 // CHECK: [[F01VALUE:%.*]] = load i64, ptr [[F01]] // CHECK: [[CASTF01:%.*]] = ptrtoint ptr %f01 to i64 // CHECK: [[BLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 [[CASTF01]], i64 1276) // CHECK: [[AUTHED:%.*]] = call i64 @llvm.ptrauth.resign(i64 [[F01VALUE]], i32 1, i64 [[BLENDED]], i32 1, i64 431) } // CHECK-LABEL: define i64 @other_test __INTPTR_TYPE__ other_test(__INTPTR_TYPE__ i) { __INTPTR_TYPE__ __ptrauth(1, 1, 42) j = 0; // CHECK: [[J_ADDR:%.*]] = ptrtoint ptr %j to i64 // CHECK: store i64 0, ptr %j __INTPTR_TYPE__ __ptrauth(1, 1, 43) k = 1234; // CHECK: [[ADDR:%.*]] = ptrtoint ptr %k to i64 // CHECK: [[JBLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 [[ADDR]], i64 43) // CHECK: [[SIGNED:%.*]] = call i64 @llvm.ptrauth.sign(i64 1234, i32 1, i64 [[JBLENDED]]) __INTPTR_TYPE__ __ptrauth(1, 1, 44) l = i; // CHECK: [[I:%.*]] = load i64, ptr %i.addr // CHECK: [[ADDR:%.*]] = ptrtoint ptr %l to i64 // CHECK: [[LBLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 [[ADDR]], i64 44) // CHECK: [[SIGNED:%.*]] = call i64 @llvm.ptrauth.sign(i64 [[I]], i32 1, i64 [[LBLENDED]]) asm volatile ("" ::: "memory"); return j + k + l; // CHECK: [[VALUE:%.*]] = load i64, ptr %j // CHECK: [[CAST_ADDR:%.*]] = ptrtoint ptr %j to i64 // CHECK: [[BLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 [[CAST_ADDR]], i64 42) // CHECK: [[AUTHED:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[VALUE]], i32 1, i64 [[BLENDED]]) // CHECK: [[VALUE:%.*]] = load i64, ptr %k // CHECK: [[CASTK:%.*]] = ptrtoint ptr %k to i64 // CHECK: [[BLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 [[CASTK]], i64 43) // CHECK: [[AUTHED:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[VALUE]], i32 1, i64 [[BLENDED]]) // CHECK: [[VALUE:%.*]] = load i64, ptr %l // CHECK: [[CASTL:%.*]] = ptrtoint ptr %l to i64 // CHECK: [[BLENDED:%.*]] = call i64 @llvm.ptrauth.blend(i64 [[CASTL]], i64 44) // CHECK: [[AUTHED:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[VALUE]], i32 1, i64 [[BLENDED]]) }