blob: 7c02d83034d604a193f694497036cce63ede5b1e [file] [edit]
; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py UTC_ARGS: --version 6
; RUN: llc -mtriple=thumbv6m-none-eabi < %s | FileCheck %s
; This test verifies that Thumb1 (ARMv6-M) generates correct code for backend KCFI.
; Thumb1 uses the backend KCFI implementation with Thumb1-specific instructions.
; Test function without KCFI annotation
; CHECK-LABEL: .globl nosan
; CHECK-NEXT: .p2align 1
; CHECK-NEXT: .type nosan,%function
; CHECK-NEXT: .code 16
; CHECK-NEXT: .thumb_func
define dso_local void @nosan() nounwind {
; CHECK-LABEL: nosan:
; CHECK: @ %bb.0:
; CHECK-NEXT: bx lr
ret void
}
; Test function with KCFI annotation - verifies type hash emission
;; The alignment is at least 4 to avoid unaligned type hash loads when this
;; instrumented function is indirectly called.
; CHECK-LABEL: .globl target_func
; CHECK-NEXT: .p2align 2
; CHECK-NEXT: .type target_func,%function
; CHECK-NEXT: .long 3170468932
; CHECK-NEXT: .code 16
; CHECK-NEXT: .thumb_func
define void @target_func() !kcfi_type !1 {
; CHECK-LABEL: target_func:
; CHECK: @ %bb.0:
; CHECK-NEXT: bx lr
ret void
}
; Test indirect call with KCFI check using operand bundles
; CHECK-LABEL: .globl f1
; CHECK: .p2align 2
; CHECK-NEXT: .type f1,%function
; CHECK-NEXT: .long 3170468932
; CHECK-NEXT: .code 16
; CHECK-NEXT: .thumb_func
define void @f1(ptr noundef %x) !kcfi_type !1 {
; CHECK-LABEL: f1:
; CHECK: @ %bb.0:
; CHECK-NEXT: .save {r7, lr}
; CHECK-NEXT: push {r7, lr}
; CHECK-NEXT: movs r3, #1
; CHECK-NEXT: mov r2, r0
; CHECK-NEXT: bics r2, r3
; CHECK-NEXT: subs r2, #4
; CHECK-NEXT: ldr r2, [r2]
; CHECK-NEXT: movs r3, #188
; CHECK-NEXT: lsls r3, r3, #8
; CHECK-NEXT: adds r3, #249
; CHECK-NEXT: lsls r3, r3, #8
; CHECK-NEXT: adds r3, #132
; CHECK-NEXT: lsls r3, r3, #8
; CHECK-NEXT: adds r3, #68
; CHECK-NEXT: cmp r2, r3
; CHECK-NEXT: beq .Ltmp0
; CHECK-NEXT: bkpt #0
; CHECK-NEXT: .Ltmp0:
; CHECK-NEXT: blx r0
; CHECK-NEXT: pop {r7, pc}
call void %x() [ "kcfi"(i32 -1124498364) ]
ret void
}
; Test with tail call - backend KCFI supports tail calls
define void @f2(ptr noundef %x) !kcfi_type !1 {
; CHECK-LABEL: f2:
; CHECK: @ %bb.0:
; CHECK-NEXT: .save {r7, lr}
; CHECK-NEXT: push {r7, lr}
; CHECK-NEXT: movs r3, #1
; CHECK-NEXT: mov r2, r0
; CHECK-NEXT: bics r2, r3
; CHECK-NEXT: subs r2, #4
; CHECK-NEXT: ldr r2, [r2]
; CHECK-NEXT: movs r3, #188
; CHECK-NEXT: lsls r3, r3, #8
; CHECK-NEXT: adds r3, #249
; CHECK-NEXT: lsls r3, r3, #8
; CHECK-NEXT: adds r3, #132
; CHECK-NEXT: lsls r3, r3, #8
; CHECK-NEXT: adds r3, #68
; CHECK-NEXT: cmp r2, r3
; CHECK-NEXT: beq .Ltmp1
; CHECK-NEXT: bkpt #0
; CHECK-NEXT: .Ltmp1:
; CHECK-NEXT: blx r0
; CHECK-NEXT: pop {r7, pc}
tail call void %x() [ "kcfi"(i32 -1124498364) ]
ret void
}
; Test with R2 live (3 arguments) - compiler shuffles args, no spilling needed
define void @f3_r2_live(ptr noundef %x, i32 %a, i32 %b, i32 %c) !kcfi_type !1 {
; CHECK-LABEL: f3_r2_live:
; CHECK: @ %bb.0:
; CHECK-NEXT: .save {r4, lr}
; CHECK-NEXT: push {r4, lr}
; CHECK-NEXT: mov r4, r0
; CHECK-NEXT: mov r0, r1
; CHECK-NEXT: mov r1, r2
; CHECK-NEXT: mov r2, r3
; CHECK-NEXT: push {r2}
; CHECK-NEXT: movs r3, #1
; CHECK-NEXT: mov r2, r4
; CHECK-NEXT: bics r2, r3
; CHECK-NEXT: subs r2, #4
; CHECK-NEXT: ldr r2, [r2]
; CHECK-NEXT: movs r3, #188
; CHECK-NEXT: lsls r3, r3, #8
; CHECK-NEXT: adds r3, #249
; CHECK-NEXT: lsls r3, r3, #8
; CHECK-NEXT: adds r3, #132
; CHECK-NEXT: lsls r3, r3, #8
; CHECK-NEXT: adds r3, #68
; CHECK-NEXT: cmp r2, r3
; CHECK-NEXT: pop {r2}
; CHECK-NEXT: beq .Ltmp2
; CHECK-NEXT: bkpt #0
; CHECK-NEXT: .Ltmp2:
; CHECK-NEXT: blx r4
; CHECK-NEXT: pop {r4, pc}
; Compiler shuffles: target→r4, c→r2, a→r0, b→r1
; R2 is live (3rd arg), so we push it, then uses R3 as temp, R2 as scratch
call void %x(i32 %a, i32 %b, i32 %c) [ "kcfi"(i32 -1124498364) ]
ret void
}
; Test with both R2 and R3 live (4 arguments) - compiler moves to r5/r4, uses R3 temp and R12 scratch
define void @f4_r2_r3_live(ptr noundef %x, i32 %a, i32 %b, i32 %c, i32 %d) !kcfi_type !1 {
; CHECK-LABEL: f4_r2_r3_live:
; CHECK: @ %bb.0:
; CHECK-NEXT: .save {r4, r5, r7, lr}
; CHECK-NEXT: push {r4, r5, r7, lr}
; CHECK-NEXT: mov r5, r3
; CHECK-NEXT: mov r4, r0
; CHECK-NEXT: ldr r3, [sp, #16]
; CHECK-NEXT: mov r0, r1
; CHECK-NEXT: mov r1, r2
; CHECK-NEXT: mov r2, r5
; CHECK-NEXT: push {r3}
; CHECK-NEXT: push {r2}
; CHECK-NEXT: movs r3, #1
; CHECK-NEXT: mov r2, r4
; CHECK-NEXT: bics r2, r3
; CHECK-NEXT: subs r2, #4
; CHECK-NEXT: ldr r2, [r2]
; CHECK-NEXT: movs r3, #188
; CHECK-NEXT: lsls r3, r3, #8
; CHECK-NEXT: adds r3, #249
; CHECK-NEXT: lsls r3, r3, #8
; CHECK-NEXT: adds r3, #132
; CHECK-NEXT: lsls r3, r3, #8
; CHECK-NEXT: adds r3, #68
; CHECK-NEXT: cmp r2, r3
; CHECK-NEXT: pop {r2}
; CHECK-NEXT: pop {r3}
; CHECK-NEXT: beq .Ltmp3
; CHECK-NEXT: bkpt #0
; CHECK-NEXT: .Ltmp3:
; CHECK-NEXT: blx r4
; CHECK-NEXT: pop {r4, r5, r7, pc}
; Compiler shuffles: r3→r5, target→r4, d→r3 (from stack), a→r0, b→r1, c→r2
; Then pushes r3 (d value), then r2, uses R3 as temp, R2 as scratch
call void %x(i32 %a, i32 %b, i32 %c, i32 %d) [ "kcfi"(i32 -1124498364) ]
ret void
}
; Test where target ends up in R12, forcing R2 as scratch, with both R2 and R3 live
; This uses inline asm to force target into R12, with 4 call arguments to make R2/R3 live
define void @f5_r12_target_r2_r3_live(i32 %a, i32 %b, i32 %c, i32 %d) !kcfi_type !1 {
; CHECK-LABEL: f5_r12_target_r2_r3_live:
; CHECK: @ %bb.0:
; CHECK-NEXT: .save {r7, lr}
; CHECK-NEXT: push {r7, lr}
; CHECK-NEXT: @APP
; CHECK-NEXT: @NO_APP
; CHECK-NEXT: push {r3}
; CHECK-NEXT: push {r2}
; CHECK-NEXT: movs r3, #1
; CHECK-NEXT: mov r2, r12
; CHECK-NEXT: bics r2, r3
; CHECK-NEXT: subs r2, #4
; CHECK-NEXT: ldr r2, [r2]
; CHECK-NEXT: movs r3, #188
; CHECK-NEXT: lsls r3, r3, #8
; CHECK-NEXT: adds r3, #249
; CHECK-NEXT: lsls r3, r3, #8
; CHECK-NEXT: adds r3, #132
; CHECK-NEXT: lsls r3, r3, #8
; CHECK-NEXT: adds r3, #68
; CHECK-NEXT: cmp r2, r3
; CHECK-NEXT: pop {r2}
; CHECK-NEXT: pop {r3}
; CHECK-NEXT: beq .Ltmp4
; CHECK-NEXT: bkpt #0
; CHECK-NEXT: .Ltmp4:
; CHECK-NEXT: blx r12
; CHECK-NEXT: pop {r7, pc}
; Use inline asm to get function pointer into R12
; With 4 arguments (r0-r3), both R2 and R3 are live
; Target in R12 means R2 is scratch, R3 is temp, and both need spilling
%target = call ptr asm "", "={r12}"()
call void %target(i32 %a, i32 %b, i32 %c, i32 %d) [ "kcfi"(i32 -1124498364) ]
ret void
}
!llvm.module.flags = !{!0}
!0 = !{i32 4, !"kcfi", i32 1}
!1 = !{i32 -1124498364}