1 ; RUN: llc -mtriple=x86_64-unknown-unknown < %s | FileCheck %s --check-prefix=ALL --check-prefix=X86_64
2 ; RUN: llc -mtriple=x86_64-unknown-unknown-gnux32 < %s | FileCheck %s --check-prefix=ALL --check-prefix=X86_64
3 ; RUN: llc -mtriple=i386-unknown-unknown < %s | FileCheck %s --check-prefix=ALL --check-prefix=X86
4 ; FIXME: Fix machine verifier issues and remove -verify-machineinstrs=0. PR39439.
5 ; RUN: llc -mtriple i386-windows-gnu -exception-model sjlj -verify-machineinstrs=0 < %s | FileCheck %s --check-prefix=SJLJ
7 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
10 ;; Checks ENDBR insertion in case of switch case statement.
11 ;; Also since the function is not internal, make sure that endbr32/64 was
12 ;; added at the beginning of the function.
13 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
21 ; X86_64-NEXT: endbr64
24 ; X86_64-NEXT: endbr64
27 %0 = select i1 undef, i8* blockaddress(@test1, %bb), i8* blockaddress(@test1, %bb6) ; <i8*> [#uses=1]
28 indirectbr i8* %0, [label %bb, label %bb6]
37 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
40 ;; Checks NOTRACK insertion in case of switch case statement.
41 ;; Check that there is no ENDBR insertion in the following case statements.
42 ;; Also since the function is not internal, ENDBR instruction should be
43 ;; added to its first basic block.
44 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
46 define i32 @test2(i32 %a) {
50 ; ALL: notrack jmp{{q|l}} *
54 %retval = alloca i32, align 4
55 %a.addr = alloca i32, align 4
56 store i32 %a, i32* %a.addr, align 4
57 %0 = load i32, i32* %a.addr, align 4
58 switch i32 %0, label %sw.default [
66 sw.bb: ; preds = %entry
67 store i32 5, i32* %retval, align 4
70 sw.bb1: ; preds = %entry
71 store i32 7, i32* %retval, align 4
74 sw.bb2: ; preds = %entry
75 store i32 2, i32* %retval, align 4
78 sw.bb3: ; preds = %entry
79 store i32 32, i32* %retval, align 4
82 sw.bb4: ; preds = %entry
83 store i32 73, i32* %retval, align 4
86 sw.default: ; preds = %entry
87 store i32 0, i32* %retval, align 4
90 return: ; preds = %sw.default, %sw.bb4, %sw.bb3, %sw.bb2, %sw.bb1, %sw.bb
91 %1 = load i32, i32* %retval, align 4
95 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
98 ;; Checks ENDBR insertion in case of indirect call instruction.
99 ;; The new instruction should be added to the called function (test6)
100 ;; although it is internal.
101 ;; Also since the function is not internal, ENDBR instruction should be
102 ;; added to its first basic block.
103 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
105 define void @test3() {
111 %f = alloca i32 (...)*, align 8
112 store i32 (...)* bitcast (i32 (i32)* @test6 to i32 (...)*), i32 (...)** %f, align 8
113 %0 = load i32 (...)*, i32 (...)** %f, align 8
114 %call = call i32 (...) %0()
118 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
121 ;; Checks ENDBR insertion in case of setjmp-like function calls.
122 ;; Also since the function is not internal, ENDBR instruction should be
123 ;; added to its first basic block.
124 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
126 @buf = internal global [5 x i8*] zeroinitializer
127 declare i8* @llvm.frameaddress(i32)
128 declare i8* @llvm.stacksave()
129 declare i32 @llvm.eh.sjlj.setjmp(i8*)
131 define i32 @test4() {
136 ; X86_64-NEXT: endbr64
138 %fp = tail call i8* @llvm.frameaddress(i32 0)
139 store i8* %fp, i8** getelementptr inbounds ([5 x i8*], [5 x i8*]* @buf, i64 0, i64 0), align 16
140 %sp = tail call i8* @llvm.stacksave()
141 store i8* %sp, i8** getelementptr inbounds ([5 x i8*], [5 x i8*]* @buf, i64 0, i64 2), align 16
142 %r = tail call i32 @llvm.eh.sjlj.setjmp(i8* bitcast ([5 x i8*]* @buf to i8*))
146 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
149 ;; Checks ENDBR insertion in case of internal function.
150 ;; Since the function is internal and its address was not taken,
151 ;; make sure that endbr32/64 was not added at the beginning of the
153 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
155 define internal i8 @test5(){
157 ; X86_64-NOT: endbr64
162 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
165 ;; Checks ENDBR insertion in case of function that its was address taken.
166 ;; Since the function's address was taken by test3() and despite being
167 ;; internal, check for added endbr32/64 at the beginning of the function.
168 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
170 define internal i32 @test6(i32 %a) {
177 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
180 ;; Checks ENDBR insertion in case of non-intrenal function.
181 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
183 define i32 @test7() {
190 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
193 ;; Checks that NO TRACK prefix is not added for indirect jumps to a jump-
194 ;; table that was created for SJLJ dispatch.
195 ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
197 declare void @_Z20function_that_throwsv()
198 declare i32 @__gxx_personality_sj0(...)
199 declare i8* @__cxa_begin_catch(i8*)
200 declare void @__cxa_end_catch()
202 define void @test8() personality i8* bitcast (i32 (...)* @__gxx_personality_sj0 to i8*) {
206 invoke void @_Z20function_that_throwsv()
207 to label %try.cont unwind label %lpad
210 %0 = landingpad { i8*, i32 }
212 %1 = extractvalue { i8*, i32 } %0, 0
213 %2 = tail call i8* @__cxa_begin_catch(i8* %1)
214 tail call void @__cxa_end_catch()
221 !llvm.module.flags = !{!0}
223 !0 = !{i32 4, !"cf-protection-branch", i32 1}