; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py UTC_ARGS: --version 5 ; RUN: llc -mtriple=thumbv8.1m.main-arm-none-eabi < %s | FileCheck %s --check-prefix=BTI ; RUN: llc -mtriple=thumbv8.1m.main-arm-none-eabi -mattr=+no-bti-at-return-twice < %s | \ ; RUN: FileCheck %s --check-prefix=NOBTI ; C source ; -------- ; jmp_buf buf; ; ; extern void bar(int x); ; ; int foo(int x) { ; if (setjmp(buf)) ; x = 0; ; else ; bar(x); ; return x; ; } @buf = global [20 x i64] zeroinitializer, align 8 define i32 @foo(i32 %x) "branch-target-enforcement" { ; BTI-LABEL: foo: ; BTI: @ %bb.0: @ %entry ; BTI-NEXT: bti ; BTI-NEXT: .save {r4, lr} ; BTI-NEXT: push {r4, lr} ; BTI-NEXT: mov r4, r0 ; BTI-NEXT: movw r0, :lower16:buf ; BTI-NEXT: movt r0, :upper16:buf ; BTI-NEXT: bl setjmp ; BTI-NEXT: bti ; BTI-NEXT: cmp r0, #0 ; BTI-NEXT: itt ne ; BTI-NEXT: movne r0, #0 ; BTI-NEXT: popne {r4, pc} ; BTI-NEXT: .LBB0_1: @ %if.else ; BTI-NEXT: mov r0, r4 ; BTI-NEXT: bl bar ; BTI-NEXT: mov r0, r4 ; BTI-NEXT: pop {r4, pc} ; ; NOBTI-LABEL: foo: ; NOBTI: @ %bb.0: @ %entry ; NOBTI-NEXT: bti ; NOBTI-NEXT: .save {r4, lr} ; NOBTI-NEXT: push {r4, lr} ; NOBTI-NEXT: mov r4, r0 ; NOBTI-NEXT: movw r0, :lower16:buf ; NOBTI-NEXT: movt r0, :upper16:buf ; NOBTI-NEXT: bl setjmp ; NOBTI-NEXT: cmp r0, #0 ; NOBTI-NEXT: itt ne ; NOBTI-NEXT: movne r0, #0 ; NOBTI-NEXT: popne {r4, pc} ; NOBTI-NEXT: .LBB0_1: @ %if.else ; NOBTI-NEXT: mov r0, r4 ; NOBTI-NEXT: bl bar ; NOBTI-NEXT: mov r0, r4 ; NOBTI-NEXT: pop {r4, pc} entry: %call = call i32 @setjmp(ptr @buf) #0 %tobool.not = icmp eq i32 %call, 0 br i1 %tobool.not, label %if.else, label %if.end if.else: ; preds = %entry call void @bar(i32 %x) br label %if.end if.end: ; preds = %entry, %if.else %x.addr.0 = phi i32 [ %x, %if.else ], [ 0, %entry ] ret i32 %x.addr.0 } ;; Check that the BL to setjmp correctly clobbers LR define i32 @baz() "branch-target-enforcement" { ; BTI-LABEL: baz: ; BTI: @ %bb.0: @ %entry ; BTI-NEXT: bti ; BTI-NEXT: .save {r7, lr} ; BTI-NEXT: push {r7, lr} ; BTI-NEXT: .pad #160 ; BTI-NEXT: sub sp, #160 ; BTI-NEXT: mov r0, sp ; BTI-NEXT: bl setjmp ; BTI-NEXT: bti ; BTI-NEXT: movs r0, #0 ; BTI-NEXT: add sp, #160 ; BTI-NEXT: pop {r7, pc} ; ; NOBTI-LABEL: baz: ; NOBTI: @ %bb.0: @ %entry ; NOBTI-NEXT: bti ; NOBTI-NEXT: .save {r7, lr} ; NOBTI-NEXT: push {r7, lr} ; NOBTI-NEXT: .pad #160 ; NOBTI-NEXT: sub sp, #160 ; NOBTI-NEXT: mov r0, sp ; NOBTI-NEXT: bl setjmp ; NOBTI-NEXT: movs r0, #0 ; NOBTI-NEXT: add sp, #160 ; NOBTI-NEXT: pop {r7, pc} entry: %outgoing_jb = alloca [20 x i64], align 8 %call = call i32 @setjmp(ptr %outgoing_jb) returns_twice ret i32 0 } declare void @bar(i32) declare i32 @setjmp(ptr) #0 attributes #0 = { returns_twice }