2018-03-17 21:29:46 +08:00
|
|
|
; RUN: llc -mtriple=x86_64-unknown-unknown -mattr=+ibt < %s | FileCheck %s --check-prefix=ALL --check-prefix=X86_64
|
|
|
|
; RUN: llc -mtriple=i386-unknown-unknown -mattr=+ibt < %s | FileCheck %s --check-prefix=ALL --check-prefix=X86
|
|
|
|
; RUN: llc -mtriple i386-windows-gnu -mattr=+ibt -exception-model sjlj < %s | FileCheck %s --check-prefix=SJLJ
|
2018-01-09 16:51:18 +08:00
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;; Test1
|
|
|
|
;; -----
|
2018-03-17 21:29:46 +08:00
|
|
|
;; Checks ENDBR insertion in case of switch case statement.
|
2018-01-09 16:51:18 +08:00
|
|
|
;; Also since the function is not internal, make sure that endbr32/64 was
|
|
|
|
;; added at the beginning of the function.
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
|
|
|
define i8 @test1(){
|
|
|
|
; ALL-LABEL: test1
|
|
|
|
; X86_64: endbr64
|
|
|
|
; X86: endbr32
|
|
|
|
; ALL: jmp{{q|l}} *
|
|
|
|
; ALL: .LBB0_1:
|
|
|
|
; X86_64-NEXT: endbr64
|
|
|
|
; X86-NEXT: endbr32
|
|
|
|
; ALL: .LBB0_2:
|
|
|
|
; X86_64-NEXT: endbr64
|
|
|
|
; X86-NEXT: endbr32
|
|
|
|
entry:
|
|
|
|
%0 = select i1 undef, i8* blockaddress(@test1, %bb), i8* blockaddress(@test1, %bb6) ; <i8*> [#uses=1]
|
|
|
|
indirectbr i8* %0, [label %bb, label %bb6]
|
|
|
|
|
|
|
|
bb: ; preds = %entry
|
|
|
|
ret i8 1
|
|
|
|
|
|
|
|
bb6: ; preds = %entry
|
|
|
|
ret i8 2
|
|
|
|
}
|
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;; Test2
|
|
|
|
;; -----
|
2018-03-17 21:29:46 +08:00
|
|
|
;; Checks NOTRACK insertion in case of switch case statement.
|
|
|
|
;; Check that there is no ENDBR insertion in the following case statements.
|
2018-01-09 16:51:18 +08:00
|
|
|
;; Also since the function is not internal, ENDBR instruction should be
|
|
|
|
;; added to its first basic block.
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
|
|
|
define i32 @test2(i32 %a) {
|
|
|
|
; ALL-LABEL: test2
|
|
|
|
; X86_64: endbr64
|
|
|
|
; X86: endbr32
|
2018-03-17 21:29:46 +08:00
|
|
|
; ALL: notrack jmp{{q|l}} *
|
2018-01-09 16:51:18 +08:00
|
|
|
; X86_64-NOT: endbr64
|
|
|
|
; X86-NOT: endbr32
|
|
|
|
entry:
|
|
|
|
%retval = alloca i32, align 4
|
|
|
|
%a.addr = alloca i32, align 4
|
|
|
|
store i32 %a, i32* %a.addr, align 4
|
|
|
|
%0 = load i32, i32* %a.addr, align 4
|
|
|
|
switch i32 %0, label %sw.default [
|
|
|
|
i32 0, label %sw.bb
|
|
|
|
i32 1, label %sw.bb1
|
|
|
|
i32 2, label %sw.bb2
|
|
|
|
i32 3, label %sw.bb3
|
|
|
|
i32 4, label %sw.bb4
|
|
|
|
]
|
|
|
|
|
|
|
|
sw.bb: ; preds = %entry
|
|
|
|
store i32 5, i32* %retval, align 4
|
|
|
|
br label %return
|
|
|
|
|
|
|
|
sw.bb1: ; preds = %entry
|
|
|
|
store i32 7, i32* %retval, align 4
|
|
|
|
br label %return
|
|
|
|
|
|
|
|
sw.bb2: ; preds = %entry
|
|
|
|
store i32 2, i32* %retval, align 4
|
|
|
|
br label %return
|
|
|
|
|
|
|
|
sw.bb3: ; preds = %entry
|
|
|
|
store i32 32, i32* %retval, align 4
|
|
|
|
br label %return
|
|
|
|
|
|
|
|
sw.bb4: ; preds = %entry
|
|
|
|
store i32 73, i32* %retval, align 4
|
|
|
|
br label %return
|
|
|
|
|
|
|
|
sw.default: ; preds = %entry
|
|
|
|
store i32 0, i32* %retval, align 4
|
|
|
|
br label %return
|
|
|
|
|
|
|
|
return: ; preds = %sw.default, %sw.bb4, %sw.bb3, %sw.bb2, %sw.bb1, %sw.bb
|
|
|
|
%1 = load i32, i32* %retval, align 4
|
|
|
|
ret i32 %1
|
|
|
|
}
|
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;; Test3
|
|
|
|
;; -----
|
|
|
|
;; Checks ENDBR insertion in case of indirect call instruction.
|
|
|
|
;; The new instruction should be added to the called function (test6)
|
|
|
|
;; although it is internal.
|
|
|
|
;; Also since the function is not internal, ENDBR instruction should be
|
|
|
|
;; added to its first basic block.
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
|
|
|
define void @test3() {
|
|
|
|
; ALL-LABEL: test3
|
|
|
|
; X86_64: endbr64
|
|
|
|
; X86: endbr32
|
|
|
|
; ALL: call{{q|l}} *
|
|
|
|
entry:
|
|
|
|
%f = alloca i32 (...)*, align 8
|
|
|
|
store i32 (...)* bitcast (i32 (i32)* @test6 to i32 (...)*), i32 (...)** %f, align 8
|
|
|
|
%0 = load i32 (...)*, i32 (...)** %f, align 8
|
|
|
|
%call = call i32 (...) %0()
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;; Test4
|
|
|
|
;; -----
|
|
|
|
;; Checks ENDBR insertion in case of setjmp-like function calls.
|
|
|
|
;; Also since the function is not internal, ENDBR instruction should be
|
|
|
|
;; added to its first basic block.
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
|
|
|
@buf = internal global [5 x i8*] zeroinitializer
|
|
|
|
declare i8* @llvm.frameaddress(i32)
|
|
|
|
declare i8* @llvm.stacksave()
|
|
|
|
declare i32 @llvm.eh.sjlj.setjmp(i8*)
|
|
|
|
|
|
|
|
define i32 @test4() {
|
|
|
|
; ALL-LABEL: test4
|
|
|
|
; X86_64: endbr64
|
|
|
|
; X86: endbr32
|
|
|
|
; ALL: .LBB3_3:
|
|
|
|
; X86_64-NEXT: endbr64
|
|
|
|
; X86-NEXT: endbr32
|
|
|
|
%fp = tail call i8* @llvm.frameaddress(i32 0)
|
|
|
|
store i8* %fp, i8** getelementptr inbounds ([5 x i8*], [5 x i8*]* @buf, i64 0, i64 0), align 16
|
|
|
|
%sp = tail call i8* @llvm.stacksave()
|
|
|
|
store i8* %sp, i8** getelementptr inbounds ([5 x i8*], [5 x i8*]* @buf, i64 0, i64 2), align 16
|
|
|
|
%r = tail call i32 @llvm.eh.sjlj.setjmp(i8* bitcast ([5 x i8*]* @buf to i8*))
|
|
|
|
ret i32 %r
|
|
|
|
}
|
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;; Test5
|
|
|
|
;; -----
|
|
|
|
;; Checks ENDBR insertion in case of internal function.
|
|
|
|
;; Since the function is internal and its address was not taken,
|
|
|
|
;; make sure that endbr32/64 was not added at the beginning of the
|
|
|
|
;; function.
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
|
|
|
define internal i8 @test5(){
|
|
|
|
; ALL-LABEL: test5
|
|
|
|
; X86_64-NOT: endbr64
|
|
|
|
; X86-NOT: endbr32
|
|
|
|
ret i8 1
|
|
|
|
}
|
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;; Test6
|
|
|
|
;; -----
|
|
|
|
;; Checks ENDBR insertion in case of function that its was address taken.
|
|
|
|
;; Since the function's address was taken by test3() and despite being
|
|
|
|
;; internal, check for added endbr32/64 at the beginning of the function.
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
|
|
|
define internal i32 @test6(i32 %a) {
|
|
|
|
; ALL-LABEL: test6
|
|
|
|
; X86_64: endbr64
|
|
|
|
; X86: endbr32
|
|
|
|
ret i32 1
|
|
|
|
}
|
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;; Test7
|
|
|
|
;; -----
|
|
|
|
;; Checks ENDBR insertion in case of non-intrenal function.
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
|
|
|
define i32 @test7() {
|
|
|
|
; ALL-LABEL: test7
|
|
|
|
; X86_64: endbr64
|
|
|
|
; X86: endbr32
|
|
|
|
ret i32 1
|
|
|
|
}
|
2018-03-17 21:29:46 +08:00
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;; Test8
|
|
|
|
;; -----
|
|
|
|
;; Checks that NO TRACK prefix is not added for indirect jumps to a jump-
|
|
|
|
;; table that was created for SJLJ dispatch.
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
|
|
|
declare void @_Z20function_that_throwsv()
|
|
|
|
declare i32 @__gxx_personality_sj0(...)
|
|
|
|
declare i8* @__cxa_begin_catch(i8*)
|
|
|
|
declare void @__cxa_end_catch()
|
|
|
|
|
|
|
|
define void @test8() personality i8* bitcast (i32 (...)* @__gxx_personality_sj0 to i8*) {
|
|
|
|
;SJLJ-LABEL: test8
|
|
|
|
;SJLJ-NOT: ds
|
|
|
|
entry:
|
|
|
|
invoke void @_Z20function_that_throwsv()
|
|
|
|
to label %try.cont unwind label %lpad
|
|
|
|
|
|
|
|
lpad:
|
|
|
|
%0 = landingpad { i8*, i32 }
|
|
|
|
catch i8* null
|
|
|
|
%1 = extractvalue { i8*, i32 } %0, 0
|
|
|
|
%2 = tail call i8* @__cxa_begin_catch(i8* %1)
|
|
|
|
tail call void @__cxa_end_catch()
|
|
|
|
br label %try.cont
|
|
|
|
|
|
|
|
try.cont:
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
!llvm.module.flags = !{!0}
|
|
|
|
|
|
|
|
!0 = !{i32 4, !"cf-protection-branch", i32 1}
|