1 ; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py
2 ; RUN: llc < %s -mtriple=sparc-unknown-linux-gnu | FileCheck %s --check-prefixes=SPARC
3 ; RUN: llc < %s -mtriple=sparc64-unknown-linux-gnu | FileCheck %s --check-prefixes=SPARC64
5 define { i128, i8 } @muloti_test(i128 %l, i128 %r) unnamed_addr #0 {
6 ; SPARC-LABEL: muloti_test:
7 ; SPARC: .cfi_startproc
8 ; SPARC-NEXT: ! %bb.0: ! %start
9 ; SPARC-NEXT: save %sp, -96, %sp
10 ; SPARC-NEXT: .cfi_def_cfa_register %fp
11 ; SPARC-NEXT: .cfi_window_save
12 ; SPARC-NEXT: .cfi_register %o7, %i7
13 ; SPARC-NEXT: mov %i3, %g2
14 ; SPARC-NEXT: mov %i2, %g4
15 ; SPARC-NEXT: umul %i2, %i5, %i2
16 ; SPARC-NEXT: rd %y, %l7
17 ; SPARC-NEXT: ld [%fp+92], %l4
18 ; SPARC-NEXT: umul %i4, %i3, %i3
19 ; SPARC-NEXT: rd %y, %o1
20 ; SPARC-NEXT: ld [%fp+96], %g3
21 ; SPARC-NEXT: umul %i5, %g2, %l3
22 ; SPARC-NEXT: rd %y, %o0
23 ; SPARC-NEXT: umul %l4, %i1, %l2
24 ; SPARC-NEXT: rd %y, %l1
25 ; SPARC-NEXT: add %i3, %i2, %i2
26 ; SPARC-NEXT: umul %i0, %g3, %i3
27 ; SPARC-NEXT: rd %y, %l6
28 ; SPARC-NEXT: add %o0, %i2, %o2
29 ; SPARC-NEXT: umul %i1, %g3, %i2
30 ; SPARC-NEXT: rd %y, %l0
31 ; SPARC-NEXT: add %i3, %l2, %i3
32 ; SPARC-NEXT: add %l0, %i3, %l2
33 ; SPARC-NEXT: addcc %i2, %l3, %l3
34 ; SPARC-NEXT: umul %g2, %g3, %i3
35 ; SPARC-NEXT: rd %y, %i2
36 ; SPARC-NEXT: addxcc %l2, %o2, %o4
37 ; SPARC-NEXT: umul %g4, %g3, %g3
38 ; SPARC-NEXT: rd %y, %l5
39 ; SPARC-NEXT: addcc %g3, %i2, %i2
40 ; SPARC-NEXT: addxcc %l5, 0, %g3
41 ; SPARC-NEXT: umul %g2, %l4, %g2
42 ; SPARC-NEXT: rd %y, %l5
43 ; SPARC-NEXT: addcc %g2, %i2, %i2
44 ; SPARC-NEXT: addxcc %l5, 0, %g2
45 ; SPARC-NEXT: addcc %g3, %g2, %g2
46 ; SPARC-NEXT: addxcc %g0, 0, %g3
47 ; SPARC-NEXT: umul %g4, %l4, %l5
48 ; SPARC-NEXT: rd %y, %o3
49 ; SPARC-NEXT: addcc %l5, %g2, %l5
50 ; SPARC-NEXT: addxcc %o3, %g3, %o3
51 ; SPARC-NEXT: addcc %l5, %l3, %g2
52 ; SPARC-NEXT: addxcc %o3, %o4, %g3
53 ; SPARC-NEXT: mov 1, %l3
54 ; SPARC-NEXT: cmp %g3, %o3
55 ; SPARC-NEXT: bcs .LBB0_2
56 ; SPARC-NEXT: mov %l3, %o4
57 ; SPARC-NEXT: ! %bb.1: ! %start
58 ; SPARC-NEXT: mov %g0, %o4
59 ; SPARC-NEXT: .LBB0_2: ! %start
60 ; SPARC-NEXT: cmp %g2, %l5
61 ; SPARC-NEXT: bcs .LBB0_4
62 ; SPARC-NEXT: mov %l3, %l5
63 ; SPARC-NEXT: ! %bb.3: ! %start
64 ; SPARC-NEXT: mov %g0, %l5
65 ; SPARC-NEXT: .LBB0_4: ! %start
66 ; SPARC-NEXT: cmp %g3, %o3
67 ; SPARC-NEXT: be .LBB0_6
69 ; SPARC-NEXT: ! %bb.5: ! %start
70 ; SPARC-NEXT: mov %o4, %l5
71 ; SPARC-NEXT: .LBB0_6: ! %start
72 ; SPARC-NEXT: cmp %g4, 0
73 ; SPARC-NEXT: bne .LBB0_8
74 ; SPARC-NEXT: mov %l3, %o3
75 ; SPARC-NEXT: ! %bb.7: ! %start
76 ; SPARC-NEXT: mov %g0, %o3
77 ; SPARC-NEXT: .LBB0_8: ! %start
78 ; SPARC-NEXT: cmp %i4, 0
79 ; SPARC-NEXT: bne .LBB0_10
80 ; SPARC-NEXT: mov %l3, %o4
81 ; SPARC-NEXT: ! %bb.9: ! %start
82 ; SPARC-NEXT: mov %g0, %o4
83 ; SPARC-NEXT: .LBB0_10: ! %start
84 ; SPARC-NEXT: cmp %o1, 0
85 ; SPARC-NEXT: bne .LBB0_12
86 ; SPARC-NEXT: mov %l3, %o1
87 ; SPARC-NEXT: ! %bb.11: ! %start
88 ; SPARC-NEXT: mov %g0, %o1
89 ; SPARC-NEXT: .LBB0_12: ! %start
90 ; SPARC-NEXT: cmp %l7, 0
91 ; SPARC-NEXT: bne .LBB0_14
92 ; SPARC-NEXT: mov %l3, %l7
93 ; SPARC-NEXT: ! %bb.13: ! %start
94 ; SPARC-NEXT: mov %g0, %l7
95 ; SPARC-NEXT: .LBB0_14: ! %start
96 ; SPARC-NEXT: cmp %o2, %o0
97 ; SPARC-NEXT: bcs .LBB0_16
98 ; SPARC-NEXT: mov %l3, %g4
99 ; SPARC-NEXT: ! %bb.15: ! %start
100 ; SPARC-NEXT: mov %g0, %g4
101 ; SPARC-NEXT: .LBB0_16: ! %start
102 ; SPARC-NEXT: cmp %l4, 0
103 ; SPARC-NEXT: bne .LBB0_18
104 ; SPARC-NEXT: mov %l3, %l4
105 ; SPARC-NEXT: ! %bb.17: ! %start
106 ; SPARC-NEXT: mov %g0, %l4
107 ; SPARC-NEXT: .LBB0_18: ! %start
108 ; SPARC-NEXT: cmp %i0, 0
109 ; SPARC-NEXT: bne .LBB0_20
110 ; SPARC-NEXT: mov %l3, %o0
111 ; SPARC-NEXT: ! %bb.19: ! %start
112 ; SPARC-NEXT: mov %g0, %o0
113 ; SPARC-NEXT: .LBB0_20: ! %start
114 ; SPARC-NEXT: cmp %l6, 0
115 ; SPARC-NEXT: bne .LBB0_22
116 ; SPARC-NEXT: mov %l3, %l6
117 ; SPARC-NEXT: ! %bb.21: ! %start
118 ; SPARC-NEXT: mov %g0, %l6
119 ; SPARC-NEXT: .LBB0_22: ! %start
120 ; SPARC-NEXT: and %o4, %o3, %o2
121 ; SPARC-NEXT: cmp %l1, 0
122 ; SPARC-NEXT: and %o0, %l4, %l4
123 ; SPARC-NEXT: bne .LBB0_24
124 ; SPARC-NEXT: mov %l3, %l1
125 ; SPARC-NEXT: ! %bb.23: ! %start
126 ; SPARC-NEXT: mov %g0, %l1
127 ; SPARC-NEXT: .LBB0_24: ! %start
128 ; SPARC-NEXT: or %o2, %o1, %o0
129 ; SPARC-NEXT: cmp %l2, %l0
130 ; SPARC-NEXT: or %l4, %l6, %l4
131 ; SPARC-NEXT: bcs .LBB0_26
132 ; SPARC-NEXT: mov %l3, %l0
133 ; SPARC-NEXT: ! %bb.25: ! %start
134 ; SPARC-NEXT: mov %g0, %l0
135 ; SPARC-NEXT: .LBB0_26: ! %start
136 ; SPARC-NEXT: or %o0, %l7, %l2
137 ; SPARC-NEXT: or %i5, %i4, %i4
138 ; SPARC-NEXT: cmp %i4, 0
139 ; SPARC-NEXT: or %l4, %l1, %l1
140 ; SPARC-NEXT: bne .LBB0_28
141 ; SPARC-NEXT: mov %l3, %i4
142 ; SPARC-NEXT: ! %bb.27: ! %start
143 ; SPARC-NEXT: mov %g0, %i4
144 ; SPARC-NEXT: .LBB0_28: ! %start
145 ; SPARC-NEXT: or %l2, %g4, %i5
146 ; SPARC-NEXT: or %i1, %i0, %i0
147 ; SPARC-NEXT: cmp %i0, 0
148 ; SPARC-NEXT: bne .LBB0_30
149 ; SPARC-NEXT: or %l1, %l0, %i0
150 ; SPARC-NEXT: ! %bb.29: ! %start
151 ; SPARC-NEXT: mov %g0, %l3
152 ; SPARC-NEXT: .LBB0_30: ! %start
153 ; SPARC-NEXT: and %l3, %i4, %i1
154 ; SPARC-NEXT: or %i1, %i0, %i0
155 ; SPARC-NEXT: or %i0, %i5, %i0
156 ; SPARC-NEXT: or %i0, %l5, %i0
157 ; SPARC-NEXT: and %i0, 1, %i4
158 ; SPARC-NEXT: mov %g3, %i0
160 ; SPARC-NEXT: restore %g0, %g2, %o1
162 ; SPARC64-LABEL: muloti_test:
163 ; SPARC64: .cfi_startproc
164 ; SPARC64-NEXT: .register %g2, #scratch
165 ; SPARC64-NEXT: .register %g3, #scratch
166 ; SPARC64-NEXT: ! %bb.0: ! %start
167 ; SPARC64-NEXT: save %sp, -176, %sp
168 ; SPARC64-NEXT: .cfi_def_cfa_register %fp
169 ; SPARC64-NEXT: .cfi_window_save
170 ; SPARC64-NEXT: .cfi_register %o7, %i7
171 ; SPARC64-NEXT: mov %g0, %o0
172 ; SPARC64-NEXT: mov %i2, %o1
173 ; SPARC64-NEXT: mov %g0, %o2
174 ; SPARC64-NEXT: call __multi3
175 ; SPARC64-NEXT: mov %i1, %o3
176 ; SPARC64-NEXT: mov %o0, %i4
177 ; SPARC64-NEXT: mov %o1, %i5
178 ; SPARC64-NEXT: mov %g0, %o0
179 ; SPARC64-NEXT: mov %i0, %o1
180 ; SPARC64-NEXT: mov %g0, %o2
181 ; SPARC64-NEXT: call __multi3
182 ; SPARC64-NEXT: mov %i3, %o3
183 ; SPARC64-NEXT: mov %o0, %l0
184 ; SPARC64-NEXT: add %o1, %i5, %i5
185 ; SPARC64-NEXT: mov %g0, %o0
186 ; SPARC64-NEXT: mov %i1, %o1
187 ; SPARC64-NEXT: mov %g0, %o2
188 ; SPARC64-NEXT: call __multi3
189 ; SPARC64-NEXT: mov %i3, %o3
190 ; SPARC64-NEXT: mov %g0, %i3
191 ; SPARC64-NEXT: mov %g0, %g2
192 ; SPARC64-NEXT: mov %g0, %g3
193 ; SPARC64-NEXT: mov %g0, %g4
194 ; SPARC64-NEXT: mov %g0, %g5
195 ; SPARC64-NEXT: add %o0, %i5, %i1
196 ; SPARC64-NEXT: cmp %i1, %o0
197 ; SPARC64-NEXT: movrnz %l0, 1, %g2
198 ; SPARC64-NEXT: movrnz %i2, 1, %g3
199 ; SPARC64-NEXT: movrnz %i0, 1, %g4
200 ; SPARC64-NEXT: movcs %xcc, 1, %i3
201 ; SPARC64-NEXT: and %g4, %g3, %i0
202 ; SPARC64-NEXT: or %i0, %g2, %i0
203 ; SPARC64-NEXT: movrnz %i4, 1, %g5
204 ; SPARC64-NEXT: or %i0, %g5, %i0
205 ; SPARC64-NEXT: or %i0, %i3, %i0
206 ; SPARC64-NEXT: srl %i0, 0, %i2
207 ; SPARC64-NEXT: mov %i1, %i0
209 ; SPARC64-NEXT: restore %g0, %o1, %o1
211 %0 = tail call { i128, i1 } @llvm.umul.with.overflow.i128(i128 %l, i128 %r) #2
212 %1 = extractvalue { i128, i1 } %0, 0
213 %2 = extractvalue { i128, i1 } %0, 1
214 %3 = zext i1 %2 to i8
215 %4 = insertvalue { i128, i8 } undef, i128 %1, 0
216 %5 = insertvalue { i128, i8 } %4, i8 %3, 1
220 ; Function Attrs: nounwind readnone speculatable
221 declare { i128, i1 } @llvm.umul.with.overflow.i128(i128, i128) #1
223 attributes #0 = { nounwind readnone uwtable }
224 attributes #1 = { nounwind readnone speculatable }
225 attributes #2 = { nounwind }