1; RUN: opt -basicaa -print-memoryssa -verify-memoryssa -analyze < %s 2>&1 | FileCheck %s
2; RUN: opt -aa-pipeline=basic-aa -passes='print<memoryssa>,verify<memoryssa>' -disable-output < %s 2>&1 | FileCheck %s
3;
4; Ensures that atomic loads count as MemoryDefs
5
6; CHECK-LABEL: define i32 @foo
7define i32 @foo(i32* %a, i32* %b) {
8; CHECK: 1 = MemoryDef(liveOnEntry)
9; CHECK-NEXT: store i32 4
10  store i32 4, i32* %a, align 4
11; CHECK: 2 = MemoryDef(1)
12; CHECK-NEXT: %1 = load atomic i32
13  %1 = load atomic i32, i32* %b acquire, align 4
14; CHECK: MemoryUse(2)
15; CHECK-NEXT: %2 = load i32
16  %2 = load i32, i32* %a, align 4
17  %3 = add i32 %1, %2
18  ret i32 %3
19}
20
21; CHECK-LABEL: define void @bar
22define void @bar(i32* %a) {
23; CHECK: MemoryUse(liveOnEntry)
24; CHECK-NEXT: load atomic i32, i32* %a unordered, align 4
25  load atomic i32, i32* %a unordered, align 4
26; CHECK: 1 = MemoryDef(liveOnEntry)
27; CHECK-NEXT: load atomic i32, i32* %a monotonic, align 4
28  load atomic i32, i32* %a monotonic, align 4
29; CHECK: 2 = MemoryDef(1)
30; CHECK-NEXT: load atomic i32, i32* %a acquire, align 4
31  load atomic i32, i32* %a acquire, align 4
32; CHECK: 3 = MemoryDef(2)
33; CHECK-NEXT: load atomic i32, i32* %a seq_cst, align 4
34  load atomic i32, i32* %a seq_cst, align 4
35  ret void
36}
37
38; CHECK-LABEL: define void @baz
39define void @baz(i32* %a) {
40; CHECK: 1 = MemoryDef(liveOnEntry)
41; CHECK-NEXT: %1 = load atomic i32
42  %1 = load atomic i32, i32* %a acquire, align 4
43; CHECK: MemoryUse(1)
44; CHECK-NEXT: %2 = load atomic i32, i32* %a unordered, align 4
45  %2 = load atomic i32, i32* %a unordered, align 4
46; CHECK: 2 = MemoryDef(1)
47; CHECK-NEXT: %3 = load atomic i32, i32* %a monotonic, align 4
48  %3 = load atomic i32, i32* %a monotonic, align 4
49  ret void
50}
51
52; CHECK-LABEL: define void @fences
53define void @fences(i32* %a) {
54; CHECK: 1 = MemoryDef(liveOnEntry)
55; CHECK-NEXT: fence acquire
56  fence acquire
57; CHECK: MemoryUse(1)
58; CHECK-NEXT: %1 = load i32, i32* %a
59  %1 = load i32, i32* %a
60
61; CHECK: 2 = MemoryDef(1)
62; CHECK-NEXT: fence release
63  fence release
64; CHECK: MemoryUse(2)
65; CHECK-NEXT: %2 = load i32, i32* %a
66  %2 = load i32, i32* %a
67
68; CHECK: 3 = MemoryDef(2)
69; CHECK-NEXT: fence acq_rel
70  fence acq_rel
71; CHECK: MemoryUse(3)
72; CHECK-NEXT: %3 = load i32, i32* %a
73  %3 = load i32, i32* %a
74
75; CHECK: 4 = MemoryDef(3)
76; CHECK-NEXT: fence seq_cst
77  fence seq_cst
78; CHECK: MemoryUse(4)
79; CHECK-NEXT: %4 = load i32, i32* %a
80  %4 = load i32, i32* %a
81  ret void
82}
83
84; CHECK-LABEL: define void @seq_cst_clobber
85define void @seq_cst_clobber(i32* noalias %a, i32* noalias %b) {
86; CHECK: 1 = MemoryDef(liveOnEntry)
87; CHECK-NEXT: %1 = load atomic i32, i32* %a monotonic, align 4
88  load atomic i32, i32* %a monotonic, align 4
89
90; CHECK: 2 = MemoryDef(1)
91; CHECK-NEXT: %2 = load atomic i32, i32* %a seq_cst, align 4
92  load atomic i32, i32* %a seq_cst, align 4
93
94; CHECK: 3 = MemoryDef(2)
95; CHECK-NEXT: load atomic i32, i32* %a monotonic, align 4
96  load atomic i32, i32* %a monotonic, align 4
97
98  ret void
99}
100
101; Ensure that AA hands us MRI_Mod on unreorderable atomic ops.
102;
103; This test is a bit implementation-specific. In particular, it depends on that
104; we pass cmpxchg-load queries to AA, without trying to reason about them on
105; our own.
106;
107; If AA gets more aggressive, we can find another way.
108;
109; CHECK-LABEL: define void @check_aa_is_sane
110define void @check_aa_is_sane(i32* noalias %a, i32* noalias %b) {
111; CHECK: 1 = MemoryDef(liveOnEntry)
112; CHECK-NEXT: cmpxchg i32* %a, i32 0, i32 1 acquire acquire
113  cmpxchg i32* %a, i32 0, i32 1 acquire acquire
114; CHECK: MemoryUse(1)
115; CHECK-NEXT: load i32, i32* %b, align 4
116  load i32, i32* %b, align 4
117
118  ret void
119}
120