File: ptrauth.c

package info (click to toggle)
swiftlang 6.0.3-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 2,519,992 kB
  • sloc: cpp: 9,107,863; ansic: 2,040,022; asm: 1,135,751; python: 296,500; objc: 82,456; f90: 60,502; lisp: 34,951; pascal: 19,946; sh: 18,133; perl: 7,482; ml: 4,937; javascript: 4,117; makefile: 3,840; awk: 3,535; xml: 914; fortran: 619; cs: 573; ruby: 573
file content (100 lines) | stat: -rw-r--r-- 4,380 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
// RUN: %clang_cc1 -triple arm64-apple-ios -fptrauth-calls -fptrauth-intrinsics -emit-llvm %s  -o - | FileCheck -check-prefix=CHECK -check-prefix=NOPCH %s
// RUN: %clang_cc1 -triple arm64-apple-ios -fptrauth-calls -fptrauth-intrinsics -emit-pch %s -o %t.ast
// RUN: %clang_cc1 -triple arm64-apple-ios -fptrauth-calls -fptrauth-intrinsics -emit-llvm -x ast -o - %t.ast | FileCheck -check-prefix=CHECK -check-prefix=PCH %s

#define FNPTRKEY 0

void (*fnptr)(void);
long discriminator;

extern void external_function(void);
// CHECK: [[EXTERNAL_FUNCTION:@.*]] = private constant { ptr, i32, i64, i64 } { ptr @external_function, i32 0, i64 0, i64 0 }, section "llvm.ptrauth", align 8
// CHECK: @fptr1 = global ptr [[EXTERNAL_FUNCTION]]
void (*fptr1)(void) = external_function;
// CHECK: @fptr2 = global ptr [[EXTERNAL_FUNCTION]]
void (*fptr2)(void) = &external_function;

// CHECK: [[SIGNED:@.*]] = private constant { ptr, i32, i64, i64 } { ptr @external_function, i32 2, i64 0, i64 26 }, section "llvm.ptrauth", align 8
// CHECK: @fptr3 = global ptr [[SIGNED]]
void (*fptr3)(void) = __builtin_ptrauth_sign_constant(&external_function, 2, 26);

// CHECK: @fptr4 = global ptr [[SIGNED:@.*]],
// CHECK: [[SIGNED]] = private constant { ptr, i32, i64, i64 } { ptr @external_function, i32 2, i64 ptrtoint (ptr @fptr4 to i64), i64 26 }, section "llvm.ptrauth", align 8
void (*fptr4)(void) = __builtin_ptrauth_sign_constant(&external_function, 2, __builtin_ptrauth_blend_discriminator(&fptr4, 26));

// CHECK-LABEL: define void @test_call()
void test_call() {
  // CHECK:      [[T0:%.*]] = load ptr, ptr @fnptr,
  // CHECK-NEXT: call void [[T0]]() [ "ptrauth"(i32 0, i64 0) ]
  fnptr();
}

// CHECK-LABEL: define void @test_direct_call()
void test_direct_call() {
  // CHECK: call void @test_call(){{$}}
  test_call();
}

void abort();
// CHECK-LABEL: define void @test_direct_builtin_call()
void test_direct_builtin_call() {
  // CHECK: call void @abort() {{#[0-9]+$}}
  abort();
}

// CHECK-LABEL: define void @test_sign_unauthenticated_peephole()
void test_sign_unauthenticated_peephole() {
  // CHECK:      [[T0:%.*]] = load ptr, ptr @fnptr,
  // CHECK-NEXT: call void [[T0]](){{$}}
  // CHECK-NEXT: ret void
  __builtin_ptrauth_sign_unauthenticated(fnptr, FNPTRKEY, 0)();
}

// This peephole doesn't kick in because it's incorrect when ABI pointer
// authentication is enabled.
// CHECK-LABEL: define void @test_auth_peephole()
void test_auth_peephole() {
  // CHECK:      [[T0:%.*]] = load ptr, ptr @fnptr,
  // CHECK-NEXT: [[T1:%.*]] = load i64, ptr @discriminator,
  // CHECK-NEXT: [[T2:%.*]] = ptrtoint ptr [[T0]] to i64
  // CHECK-NEXT: [[T3:%.*]] = call i64 @llvm.ptrauth.auth(i64 [[T2]], i32 0, i64 [[T1]])
  // CHECK-NEXT: [[T4:%.*]] = inttoptr  i64 [[T3]] to ptr
  // CHECK-NEXT: call void [[T4]]() [ "ptrauth"(i32 0, i64 0) ]
  // CHECK-NEXT: ret void
  __builtin_ptrauth_auth(fnptr, 0, discriminator)();
}

// CHECK-LABEL: define void @test_auth_and_resign_peephole()
void test_auth_and_resign_peephole() {
  // CHECK:      [[T0:%.*]] = load ptr, ptr @fnptr,
  // CHECK-NEXT: [[T1:%.*]] = load i64, ptr @discriminator,
  // CHECK-NEXT: call void [[T0]]() [ "ptrauth"(i32 2, i64 [[T1]]) ]
  // CHECK-NEXT: ret void
  __builtin_ptrauth_auth_and_resign(fnptr, 2, discriminator, FNPTRKEY, 0)();
}

// CHECK-LABEL: define ptr @test_function_pointer()
// CHECK:        [[EXTERNAL_FUNCTION]]
void (*test_function_pointer())(void) {
  return external_function;
}

// rdar://34562484 - Handle IR types changing in the caching mechanism.
struct InitiallyIncomplete;
extern struct InitiallyIncomplete returns_initially_incomplete(void);
// CHECK-LABEL: define void @use_while_incomplete()
void use_while_incomplete() {
  // NOPCH:      [[VAR:%.*]] = alloca ptr,
  // NOPCH-NEXT: store ptr @returns_initially_incomplete.ptrauth, ptr [[VAR]],
  // PCH:        [[VAR:%.*]] = alloca ptr,
  // PCH-NEXT:   store ptr @returns_initially_incomplete.ptrauth, ptr [[VAR]],
  struct InitiallyIncomplete (*fnptr)(void) = &returns_initially_incomplete;
}
struct InitiallyIncomplete { int x; };
// CHECK-LABEL: define void @use_while_complete()
void use_while_complete() {
  // CHECK:      [[VAR:%.*]] = alloca ptr,
  // CHECK-NEXT: store ptr @returns_initially_incomplete.ptrauth, ptr [[VAR]],
  // CHECK-NEXT: ret void
  struct InitiallyIncomplete (*fnptr)(void) = &returns_initially_incomplete;
}