1; Test insertions of i32s into the low half of an i64.
2;
3; RUN: llc < %s -mtriple=s390x-linux-gnu | FileCheck %s
4
5; Insertion of an i32 can be done using LR.
6define i64 @f1(i64 %a, i32 %b) {
7; CHECK-LABEL: f1:
8; CHECK-NOT: {{%r[23]}}
9; CHECK: lr %r2, %r3
10; CHECK: br %r14
11  %low = zext i32 %b to i64
12  %high = and i64 %a, -4294967296
13  %res = or i64 %high, %low
14  ret i64 %res
15}
16
17; ... and again with the operands reversed.
18define i64 @f2(i64 %a, i32 %b) {
19; CHECK-LABEL: f2:
20; CHECK-NOT: {{%r[23]}}
21; CHECK: lr %r2, %r3
22; CHECK: br %r14
23  %low = zext i32 %b to i64
24  %high = and i64 %a, -4294967296
25  %res = or i64 %low, %high
26  ret i64 %res
27}
28
29; Like f1, but with "in register" zero extension.
30define i64 @f3(i64 %a, i64 %b) {
31; CHECK-LABEL: f3:
32; CHECK-NOT: {{%r[23]}}
33; CHECK: lr %r2, %r3
34; CHECK: br %r14
35  %low = and i64 %b, 4294967295
36  %high = and i64 %a, -4294967296
37  %res = or i64 %high, %low
38  ret i64 %res
39}
40
41; ... and again with the operands reversed.
42define i64 @f4(i64 %a, i64 %b) {
43; CHECK-LABEL: f4:
44; CHECK-NOT: {{%r[23]}}
45; CHECK: lr %r2, %r3
46; CHECK: br %r14
47  %low = and i64 %b, 4294967295
48  %high = and i64 %a, -4294967296
49  %res = or i64 %low, %high
50  ret i64 %res
51}
52
53; Unary operations can be done directly into the low half.
54define i64 @f5(i64 %a, i32 %b) {
55; CHECK-LABEL: f5:
56; CHECK-NOT: {{%r[23]}}
57; CHECK: lcr %r2, %r3
58; CHECK: br %r14
59  %neg = sub i32 0, %b
60  %low = zext i32 %neg to i64
61  %high = and i64 %a, -4294967296
62  %res = or i64 %high, %low
63  ret i64 %res
64}
65
66; ...likewise three-operand binary operations like RLL.
67define i64 @f6(i64 %a, i32 %b) {
68; CHECK-LABEL: f6:
69; CHECK-NOT: {{%r[23]}}
70; CHECK: rll %r2, %r3, 1
71; CHECK: br %r14
72  %parta = shl i32 %b, 1
73  %partb = lshr i32 %b, 31
74  %rot = or i32 %parta, %partb
75  %low = zext i32 %rot to i64
76  %high = and i64 %a, -4294967296
77  %res = or i64 %low, %high
78  ret i64 %res
79}
80
81; Loads can be done directly into the low half.  The range of L is checked
82; in the move tests.
83define i64 @f7(i64 %a, i32 *%src) {
84; CHECK-LABEL: f7:
85; CHECK-NOT: {{%r[23]}}
86; CHECK: l %r2, 0(%r3)
87; CHECK: br %r14
88  %b = load i32 , i32 *%src
89  %low = zext i32 %b to i64
90  %high = and i64 %a, -4294967296
91  %res = or i64 %high, %low
92  ret i64 %res
93}
94
95; ...likewise extending loads.
96define i64 @f8(i64 %a, i8 *%src) {
97; CHECK-LABEL: f8:
98; CHECK-NOT: {{%r[23]}}
99; CHECK: lb %r2, 0(%r3)
100; CHECK: br %r14
101  %byte = load i8 , i8 *%src
102  %b = sext i8 %byte to i32
103  %low = zext i32 %b to i64
104  %high = and i64 %a, -4294967296
105  %res = or i64 %high, %low
106  ret i64 %res
107}
108
109; Check a case like f1 in which there is no AND.  We simply know from context
110; that the upper half of one OR operand and the lower half of the other are
111; both clear.
112define i64 @f9(i64 %a, i32 %b) {
113; CHECK-LABEL: f9:
114; CHECK: sllg %r2, %r2, 32
115; CHECK: lr %r2, %r3
116; CHECK: br %r14
117  %shift = shl i64 %a, 32
118  %low = zext i32 %b to i64
119  %or = or i64 %shift, %low
120  ret i64 %or
121}
122
123; ...and again with the operands reversed.
124define i64 @f10(i64 %a, i32 %b) {
125; CHECK-LABEL: f10:
126; CHECK: sllg %r2, %r2, 32
127; CHECK: lr %r2, %r3
128; CHECK: br %r14
129  %shift = shl i64 %a, 32
130  %low = zext i32 %b to i64
131  %or = or i64 %low, %shift
132  ret i64 %or
133}
134
135; Like f9, but with "in register" zero extension.
136define i64 @f11(i64 %a, i64 %b) {
137; CHECK-LABEL: f11:
138; CHECK: lr %r2, %r3
139; CHECK: br %r14
140  %shift = shl i64 %a, 32
141  %low = and i64 %b, 4294967295
142  %or = or i64 %shift, %low
143  ret i64 %or
144}
145
146; ...and again with the operands reversed.
147define i64 @f12(i64 %a, i64 %b) {
148; CHECK-LABEL: f12:
149; CHECK: lr %r2, %r3
150; CHECK: br %r14
151  %shift = shl i64 %a, 32
152  %low = and i64 %b, 4294967295
153  %or = or i64 %low, %shift
154  ret i64 %or
155}
156
157; Like f9, but for larger shifts than 32.
158define i64 @f13(i64 %a, i32 %b) {
159; CHECK-LABEL: f13:
160; CHECK: sllg %r2, %r2, 60
161; CHECK: lr %r2, %r3
162; CHECK: br %r14
163  %shift = shl i64 %a, 60
164  %low = zext i32 %b to i64
165  %or = or i64 %shift, %low
166  ret i64 %or
167}
168
169; We previously wrongly removed the upper AND as dead.
170define i64 @f14(i64 %a, i64 %b) {
171; CHECK-LABEL: f14:
172; CHECK: risbg {{%r[0-5]}}, %r2, 6, 134, 0
173; CHECK: br %r14
174  %and1 = and i64 %a, 144115188075855872
175  %and2 = and i64 %b, 15
176  %or = or i64 %and1, %and2
177  %res = icmp eq i64 %or, 0
178  %ext = sext i1 %res to i64
179  ret i64 %ext
180}
181
182; Check another representation of f8.
183define i64 @f15(i64 %a, i8 *%src) {
184; CHECK-LABEL: f15:
185; CHECK-NOT: {{%r[23]}}
186; CHECK: lb %r2, 0(%r3)
187; CHECK: br %r14
188  %byte = load i8 , i8 *%src
189  %b = sext i8 %byte to i64
190  %low = and i64 %b, 4294967295
191  %high = and i64 %a, -4294967296
192  %res = or i64 %high, %low
193  ret i64 %res
194}
195