// This test checks that the foo function having exclusive memory access
// instructions won't be instrumented.
// REQUIRES: system-linux,bolt-runtime,target=aarch64{{.*}}
// RUN: llvm-mc -filetype=obj -triple aarch64-unknown-unknown \
// RUN: %s -o %t.o
// RUN: %clang %cflags -fPIC -pie %t.o -o %t.exe -nostdlib -Wl,-q -Wl,-fini=dummy
// RUN: llvm-bolt %t.exe -o %t.bolt -instrument -v=2 | FileCheck %s
// CHECK: BOLT-INSTRUMENTER: skip BB {{.*}} due to exclusive instruction in function foo
// CHECK: BOLT-INSTRUMENTER: skip BB {{.*}} due to exclusive instruction in function foo
// CHECK: BOLT-INSTRUMENTER: skip BB {{.*}} due to exclusive instruction in function foo
// CHECK: BOLT-INSTRUMENTER: skip BB {{.*}} due to exclusive instruction in function case1
// CHECK: BOLT-INSTRUMENTER: skip BB {{.*}} due to exclusive instruction in function case2
// CHECK: BOLT-INSTRUMENTER: skip BB {{.*}} due to exclusive instruction in function case2
// CHECK: BOLT-INSTRUMENTER: function case3 has exclusive store without corresponding load. Ignoring the function.
// CHECK: BOLT-INSTRUMENTER: skip BB {{.*}} due to exclusive instruction in function case4
// CHECK: BOLT-INSTRUMENTER: function case4 has two exclusive loads. Ignoring the function.
// CHECK: BOLT-INSTRUMENTER: skip BB {{.*}} due to exclusive instruction in function case5
// CHECK: BOLT-INSTRUMENTER: function case5 has exclusive load in trailing BB. Ignoring the function.
.global foo
.type foo, %function
foo:
# exclusive load and store in two bbs
ldaxr w9, [x10]
cbnz w9, .Lret
stlxr w12, w11, [x9]
cbz w12, foo
.Lret:
clrex
ret
.size foo, .-foo
.global _start
.type _start, %function
_start:
mov x0, #0
mov x1, #1
mov x2, #2
mov x3, #3
bl case1
bl case2
bl case3
bl case4
bl case5
ret
.size _start, .-_start
# Case 1: exclusive load and store in one basic block
.global case1
.type case1, %function
case1:
str x0, [x2]
ldxr w0, [x2]
add w0, w0, #1
stxr w1, w0, [x2]
ret
.size case1, .-case1
# Case 2: exclusive load and store in different blocks
.global case2
.type case2, %function
case2:
b case2_load
case2_load:
ldxr x0, [x2]
b case2_store
case2_store:
add x0, x0, #1
stxr w1, x0, [x2]
ret
.size case2, .-case2
# Case 3: store without preceding load
.global case3
.type case3, %function
case3:
stxr w1, x3, [x2]
ret
.size case3, .-case3
# Case 4: two exclusive load instructions in neighboring blocks
.global case4
.type case4, %function
case4:
b case4_load
case4_load:
ldxr x0, [x2]
b case4_load_next
case4_load_next:
ldxr x1, [x2]
ret
.size case4, .-case4
# Case 5: Exclusive load without successor
.global case5
.type case5, %function
case5:
ldxr x0, [x2]
ret
.size case5, .-case5
.global dummy
.type dummy, %function
dummy:
ret
.size dummy, .-dummy