|
| 1 | +; RUN: llc -mtriple=armv7-linux-gnueabi -verify-machineinstrs < %s | FileCheck %s --check-prefix=ASM |
| 2 | +; RUN: llc -mtriple=armv7-linux-gnueabi -verify-machineinstrs -stop-after=finalize-isel < %s | FileCheck %s --check-prefixes=MIR,ISEL |
| 3 | +; RUN: llc -mtriple=armv7-linux-gnueabi -verify-machineinstrs -stop-after=kcfi < %s | FileCheck %s --check-prefixes=MIR,KCFI |
| 4 | + |
| 5 | +; ASM: .long 12345678 |
| 6 | +define void @f1(ptr noundef %x) !kcfi_type !1 { |
| 7 | +; ASM-LABEL: f1: |
| 8 | +; ASM: @ %bb.0: |
| 9 | +; ASM: bic r12, r0, #1 |
| 10 | +; ASM-NEXT: ldr r12, [r12, #-4] |
| 11 | +; ASM-NEXT: eor r12, r12, #78 |
| 12 | +; ASM-NEXT: eor r12, r12, #24832 |
| 13 | +; ASM-NEXT: eor r12, r12, #12320768 |
| 14 | +; ASM-NEXT: eors r12, r12, #0 |
| 15 | +; ASM-NEXT: beq .Ltmp{{[0-9]+}} |
| 16 | +; UDF encoding: 0x8000 | (0x1F << 5) | r0 = 0x83e0 = 33760 |
| 17 | +; ASM-NEXT: udf #33760 |
| 18 | +; ASM-NEXT: .Ltmp{{[0-9]+}}: |
| 19 | +; ASM-NEXT: blx r0 |
| 20 | + |
| 21 | +; MIR-LABEL: name: f1 |
| 22 | +; MIR: body: |
| 23 | + |
| 24 | +; ISEL: BLX %0, csr_aapcs,{{.*}} cfi-type 12345678 |
| 25 | + |
| 26 | +; KCFI: BUNDLE{{.*}} { |
| 27 | +; KCFI-NEXT: KCFI_CHECK $r0, 12345678 |
| 28 | +; KCFI-NEXT: BLX killed $r0, csr_aapcs,{{.*}} |
| 29 | +; KCFI-NEXT: } |
| 30 | + |
| 31 | + call void %x() [ "kcfi"(i32 12345678) ] |
| 32 | + ret void |
| 33 | +} |
| 34 | + |
| 35 | +; Test with tail call |
| 36 | +define void @f2(ptr noundef %x) !kcfi_type !1 { |
| 37 | +; ASM-LABEL: f2: |
| 38 | +; ASM: @ %bb.0: |
| 39 | +; ASM: bic r12, r0, #1 |
| 40 | +; ASM: ldr r12, [r12, #-4] |
| 41 | +; ASM: eor r12, r12, #78 |
| 42 | +; ASM: eor r12, r12, #24832 |
| 43 | +; ASM: eor r12, r12, #12320768 |
| 44 | +; ASM: eors r12, r12, #0 |
| 45 | +; ASM: beq .Ltmp{{[0-9]+}} |
| 46 | +; UDF encoding: 0x8000 | (0x1F << 5) | r0 = 0x83e0 = 33760 |
| 47 | +; ASM: udf #33760 |
| 48 | +; ASM: .Ltmp{{[0-9]+}}: |
| 49 | +; ASM: bx r0 |
| 50 | + |
| 51 | +; MIR-LABEL: name: f2 |
| 52 | +; MIR: body: |
| 53 | + |
| 54 | +; ISEL: TCRETURNri %0, 0, csr_aapcs, implicit $sp, cfi-type 12345678 |
| 55 | + |
| 56 | +; KCFI: BUNDLE{{.*}} { |
| 57 | +; KCFI-NEXT: KCFI_CHECK $r0, 12345678 |
| 58 | +; KCFI-NEXT: TAILJMPr killed $r0, csr_aapcs, implicit $sp, implicit $sp |
| 59 | +; KCFI-NEXT: } |
| 60 | + |
| 61 | + tail call void %x() [ "kcfi"(i32 12345678) ] |
| 62 | + ret void |
| 63 | +} |
| 64 | + |
| 65 | +; Test r3 spill/reload when target is r12 and r3 is a call argument. |
| 66 | +; With 5+ arguments (target + 4 args), r0-r3 are all used for arguments, |
| 67 | +; forcing r3 to be spilled when we need it as scratch register. |
| 68 | +define void @f3_r3_spill(ptr noundef %target, i32 %a, i32 %b, i32 %c, i32 %d) !kcfi_type !1 { |
| 69 | +; ASM-LABEL: f3_r3_spill: |
| 70 | +; ASM: @ %bb.0: |
| 71 | +; Arguments: r0=%target, r1=%a, r2=%b, r3=%c, [sp]=%d |
| 72 | +; Call needs: r0=%a, r1=%b, r2=%c, r3=%d, target in r12 |
| 73 | +; Compiler shuffles arguments into place, saving r3 (c) in lr, loading d from stack |
| 74 | +; ASM: mov lr, r3 |
| 75 | +; ASM-NEXT: ldr r3, [sp, #8] |
| 76 | +; ASM-NEXT: mov r12, r0 |
| 77 | +; ASM-NEXT: mov r0, r1 |
| 78 | +; ASM-NEXT: mov r1, r2 |
| 79 | +; ASM-NEXT: mov r2, lr |
| 80 | +; r3 is live as 4th argument, so push it before KCFI check |
| 81 | +; ASM-NEXT: stmdb sp!, {r3} |
| 82 | +; ASM-NEXT: bic r3, r12, #1 |
| 83 | +; ASM-NEXT: ldr r3, [r3, #-4] |
| 84 | +; ASM-NEXT: eor r3, r3, #78 |
| 85 | +; ASM-NEXT: eor r3, r3, #24832 |
| 86 | +; ASM-NEXT: eor r3, r3, #12320768 |
| 87 | +; ASM-NEXT: eors r3, r3, #0 |
| 88 | +; Restore r3 immediately after comparison, before branch |
| 89 | +; ASM-NEXT: ldm sp!, {r3} |
| 90 | +; ASM-NEXT: beq .Ltmp{{[0-9]+}} |
| 91 | +; UDF encoding: 0x8000 | (0x1F << 5) | r12 = 0x83ec = 33772 |
| 92 | +; ASM-NEXT: udf #33772 |
| 93 | +; ASM-NEXT: .Ltmp{{[0-9]+}}: |
| 94 | +; ASM-NEXT: blx r12 |
| 95 | +; |
| 96 | + call void %target(i32 %a, i32 %b, i32 %c, i32 %d) [ "kcfi"(i32 12345678) ] |
| 97 | + ret void |
| 98 | +} |
| 99 | + |
| 100 | +; Test with 3 arguments - r3 not live, target in r12, so r3 used as scratch without spilling |
| 101 | +define void @f4_r3_unused(ptr noundef %target, i32 %a, i32 %b) !kcfi_type !1 { |
| 102 | +; ASM-LABEL: f4_r3_unused: |
| 103 | +; ASM: @ %bb.0: |
| 104 | +; Only 3 arguments total, so r3 is not used as call argument |
| 105 | +; Compiler puts target→r3, a→r0, b→r1 |
| 106 | +; ASM: mov r3, r0 |
| 107 | +; ASM-NEXT: mov r0, r1 |
| 108 | +; ASM-NEXT: mov r1, r2 |
| 109 | +; r3 is the target, so we use r12 as scratch (no spill needed) |
| 110 | +; ASM-NEXT: bic r12, r3, #1 |
| 111 | +; ASM-NEXT: ldr r12, [r12, #-4] |
| 112 | +; ASM-NEXT: eor r12, r12, #78 |
| 113 | +; ASM-NEXT: eor r12, r12, #24832 |
| 114 | +; ASM-NEXT: eor r12, r12, #12320768 |
| 115 | +; ASM-NEXT: eors r12, r12, #0 |
| 116 | +; ASM-NEXT: beq .Ltmp{{[0-9]+}} |
| 117 | +; UDF encoding: 0x8000 | (0x1F << 5) | r3 = 0x83e3 = 33763 |
| 118 | +; ASM-NEXT: udf #33763 |
| 119 | +; ASM-NEXT: .Ltmp{{[0-9]+}}: |
| 120 | +; ASM-NEXT: blx r3 |
| 121 | +; |
| 122 | + call void %target(i32 %a, i32 %b) [ "kcfi"(i32 12345678) ] |
| 123 | + ret void |
| 124 | +} |
| 125 | + |
| 126 | +!llvm.module.flags = !{!0} |
| 127 | +!0 = !{i32 4, !"kcfi", i32 1} |
| 128 | +!1 = !{i32 12345678} |
0 commit comments