1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
|
// RUN: mlir-opt -allow-unregistered-dialect -convert-scf-to-emitc %s | FileCheck %s
// RUN: mlir-opt -allow-unregistered-dialect -convert-to-emitc="filter-dialects=scf" %s | FileCheck %s
emitc.func @payload_one_result(%arg: i32) -> i32 {
%result = add %arg, %arg : (i32, i32) -> i32
return %result : i32
}
func.func @one_result() -> i32 {
%init = emitc.literal "1.0" : i32
%var = emitc.literal "1.0" : i32
%exit = emitc.literal "10.0" : i32
%res = scf.while (%arg1 = %init) : (i32) -> i32 {
%sum = emitc.add %arg1, %var : (i32, i32) -> i32
%condition = emitc.cmp lt, %sum, %exit : (i32, i32) -> i1
%next = emitc.add %arg1, %arg1 : (i32, i32) -> i32
scf.condition(%condition) %next : i32
} do {
^bb0(%arg2: i32):
%next_arg1 = emitc.call @payload_one_result(%arg2) : (i32) -> i32
scf.yield %next_arg1 : i32
}
return %res : i32
}
// CHECK-LABEL: emitc.func @payload_one_result(
// CHECK-SAME: %[[ARG0:.*]]: i32) -> i32 {
// CHECK: %[[VAL_0:.*]] = add %[[ARG0]], %[[ARG0]] : (i32, i32) -> i32
// CHECK: return %[[VAL_0]] : i32
// CHECK: }
// CHECK-LABEL: func.func @one_result() -> i32 {
// CHECK: %[[VAL_0:.*]] = emitc.literal "1.0" : i32
// CHECK: %[[VAL_1:.*]] = emitc.literal "1.0" : i32
// CHECK: %[[VAL_2:.*]] = emitc.literal "10.0" : i32
// CHECK: %[[VAL_3:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i32>
// CHECK: %[[VAL_4:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i32>
// CHECK: emitc.assign %[[VAL_0]] : i32 to %[[VAL_4]] : <i32>
// CHECK: %[[VAL_5:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i1>
// CHECK: emitc.do {
// CHECK: %[[VAL_6:.*]] = load %[[VAL_4]] : <i32>
// CHECK: %[[VAL_7:.*]] = add %[[VAL_6]], %[[VAL_1]] : (i32, i32) -> i32
// CHECK: %[[VAL_8:.*]] = cmp lt, %[[VAL_7]], %[[VAL_2]] : (i32, i32) -> i1
// CHECK: %[[VAL_9:.*]] = add %[[VAL_6]], %[[VAL_6]] : (i32, i32) -> i32
// CHECK: assign %[[VAL_9]] : i32 to %[[VAL_3]] : <i32>
// CHECK: assign %[[VAL_8]] : i1 to %[[VAL_5]] : <i1>
// CHECK: if %[[VAL_8]] {
// CHECK: %[[VAL_10:.*]] = call @payload_one_result(%[[VAL_9]]) : (i32) -> i32
// CHECK: assign %[[VAL_10]] : i32 to %[[VAL_4]] : <i32>
// CHECK: }
// CHECK: } while {
// CHECK: %[[VAL_11:.*]] = expression %[[VAL_5]] : (!emitc.lvalue<i1>) -> i1 {
// CHECK: %[[VAL_12:.*]] = load %[[VAL_5]] : <i1>
// CHECK: yield %[[VAL_12]] : i1
// CHECK: }
// CHECK: yield %[[VAL_11]] : i1
// CHECK: }
// CHECK: %[[VAL_13:.*]] = emitc.load %[[VAL_3]] : <i32>
// CHECK: return %[[VAL_13]] : i32
// CHECK: }
emitc.func @payload_two_results(%arg: i32) -> i32 {
%result = add %arg, %arg : (i32, i32) -> i32
return %result : i32
}
func.func @two_results() -> i32 {
%init = emitc.literal "1.0" : i32
%exit = emitc.literal "10.0" : i32
%res1, %res2 = scf.while (%arg1_1 = %init, %arg1_2 = %init) : (i32, i32) -> (i32, i32) {
%sum = emitc.add %arg1_1, %arg1_2 : (i32, i32) -> i32
%condition = emitc.cmp lt, %sum, %exit : (i32, i32) -> i1
scf.condition(%condition) %init, %arg1_2 : i32, i32
} do {
^bb0(%arg2_1 : i32, %arg2_2 : i32):
%next1 = emitc.call @payload_two_results(%arg2_1) : (i32) -> i32
%next2 = emitc.call @payload_two_results(%arg2_2) : (i32) -> i32
scf.yield %next1, %next2 : i32, i32
}
return %res1 : i32
}
// CHECK-LABEL: emitc.func @payload_two_results(
// CHECK-SAME: %[[ARG0:.*]]: i32) -> i32 {
// CHECK: %[[VAL_0:.*]] = add %[[ARG0]], %[[ARG0]] : (i32, i32) -> i32
// CHECK: return %[[VAL_0]] : i32
// CHECK: }
// CHECK-LABEL: func.func @two_results() -> i32 {
// CHECK: %[[VAL_0:.*]] = emitc.literal "1.0" : i32
// CHECK: %[[VAL_1:.*]] = emitc.literal "10.0" : i32
// CHECK: %[[VAL_2:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i32>
// CHECK: %[[VAL_3:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i32>
// CHECK: %[[VAL_4:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i32>
// CHECK: emitc.assign %[[VAL_0]] : i32 to %[[VAL_4]] : <i32>
// CHECK: %[[VAL_5:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i32>
// CHECK: emitc.assign %[[VAL_0]] : i32 to %[[VAL_5]] : <i32>
// CHECK: %[[VAL_6:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i1>
// CHECK: emitc.do {
// CHECK: %[[VAL_7:.*]] = load %[[VAL_4]] : <i32>
// CHECK: %[[VAL_8:.*]] = load %[[VAL_5]] : <i32>
// CHECK: %[[VAL_9:.*]] = add %[[VAL_7]], %[[VAL_8]] : (i32, i32) -> i32
// CHECK: %[[VAL_10:.*]] = cmp lt, %[[VAL_9]], %[[VAL_1]] : (i32, i32) -> i1
// CHECK: assign %[[VAL_0]] : i32 to %[[VAL_2]] : <i32>
// CHECK: assign %[[VAL_8]] : i32 to %[[VAL_3]] : <i32>
// CHECK: assign %[[VAL_10]] : i1 to %[[VAL_6]] : <i1>
// CHECK: if %[[VAL_10]] {
// CHECK: %[[VAL_11:.*]] = call @payload_two_results(%[[VAL_0]]) : (i32) -> i32
// CHECK: %[[VAL_12:.*]] = call @payload_two_results(%[[VAL_8]]) : (i32) -> i32
// CHECK: assign %[[VAL_11]] : i32 to %[[VAL_4]] : <i32>
// CHECK: assign %[[VAL_12]] : i32 to %[[VAL_5]] : <i32>
// CHECK: }
// CHECK: } while {
// CHECK: %[[VAL_13:.*]] = expression %[[VAL_6]] : (!emitc.lvalue<i1>) -> i1 {
// CHECK: %[[VAL_14:.*]] = load %[[VAL_6]] : <i1>
// CHECK: yield %[[VAL_14]] : i1
// CHECK: }
// CHECK: yield %[[VAL_13]] : i1
// CHECK: }
// CHECK: %[[VAL_15:.*]] = emitc.load %[[VAL_2]] : <i32>
// CHECK: %[[VAL_16:.*]] = emitc.load %[[VAL_3]] : <i32>
// CHECK: return %[[VAL_15]] : i32
// CHECK: }
emitc.func @payload_double_use(%arg: i32) -> i32 {
%result = add %arg, %arg : (i32, i32) -> i32
return %result : i32
}
emitc.func @foo_with_side_effect(%arg: i32, %p : !emitc.ptr<i32>) -> i32 {
%sum = add %arg, %arg : (i32, i32) -> i32
emitc.verbatim "{}[0] = {};" args %p, %sum : !emitc.ptr<i32>, i32
return %sum : i32
}
func.func @double_use(%p : !emitc.ptr<i32>) -> i32 {
%init = emitc.literal "1.0" : i32
%var = emitc.literal "1.0" : i32
%exit = emitc.literal "10.0" : i32
%res = scf.while (%arg1 = %init) : (i32) -> i32 {
%used_twice = emitc.call @foo_with_side_effect(%arg1, %p) : (i32, !emitc.ptr<i32>) -> i32
%prod = emitc.add %used_twice, %used_twice : (i32, i32) -> i32
%sum = emitc.add %arg1, %prod : (i32, i32) -> i32
%condition = emitc.cmp lt, %sum, %exit : (i32, i32) -> i1
scf.condition(%condition) %arg1 : i32
} do {
^bb0(%arg2: i32):
%next_arg1 = emitc.call @payload_double_use(%arg2) : (i32) -> i32
scf.yield %next_arg1 : i32
}
return %res : i32
}
// CHECK-LABEL: emitc.func @payload_double_use(
// CHECK-SAME: %[[ARG0:.*]]: i32) -> i32 {
// CHECK: %[[VAL_0:.*]] = add %[[ARG0]], %[[ARG0]] : (i32, i32) -> i32
// CHECK: return %[[VAL_0]] : i32
// CHECK: }
// CHECK-LABEL: emitc.func @foo_with_side_effect(
// CHECK-SAME: %[[ARG0:.*]]: i32,
// CHECK-SAME: %[[ARG1:.*]]: !emitc.ptr<i32>) -> i32 {
// CHECK: %[[VAL_0:.*]] = add %[[ARG0]], %[[ARG0]] : (i32, i32) -> i32
// CHECK: verbatim "{}[0] = {};" args %[[ARG1]], %[[VAL_0]] : !emitc.ptr<i32>, i32
// CHECK: return %[[VAL_0]] : i32
// CHECK: }
// CHECK-LABEL: func.func @double_use(
// CHECK-SAME: %[[ARG0:.*]]: !emitc.ptr<i32>) -> i32 {
// CHECK: %[[VAL_0:.*]] = emitc.literal "1.0" : i32
// CHECK: %[[VAL_1:.*]] = emitc.literal "1.0" : i32
// CHECK: %[[VAL_2:.*]] = emitc.literal "10.0" : i32
// CHECK: %[[VAL_3:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i32>
// CHECK: %[[VAL_4:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i32>
// CHECK: emitc.assign %[[VAL_0]] : i32 to %[[VAL_4]] : <i32>
// CHECK: %[[VAL_5:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i1>
// CHECK: emitc.do {
// CHECK: %[[VAL_6:.*]] = load %[[VAL_4]] : <i32>
// CHECK: %[[VAL_7:.*]] = call @foo_with_side_effect(%[[VAL_6]], %[[ARG0]]) : (i32, !emitc.ptr<i32>) -> i32
// CHECK: %[[VAL_8:.*]] = add %[[VAL_7]], %[[VAL_7]] : (i32, i32) -> i32
// CHECK: %[[VAL_9:.*]] = add %[[VAL_6]], %[[VAL_8]] : (i32, i32) -> i32
// CHECK: %[[VAL_10:.*]] = cmp lt, %[[VAL_9]], %[[VAL_2]] : (i32, i32) -> i1
// CHECK: assign %[[VAL_6]] : i32 to %[[VAL_3]] : <i32>
// CHECK: assign %[[VAL_10]] : i1 to %[[VAL_5]] : <i1>
// CHECK: if %[[VAL_10]] {
// CHECK: %[[VAL_11:.*]] = call @payload_double_use(%[[VAL_6]]) : (i32) -> i32
// CHECK: assign %[[VAL_11]] : i32 to %[[VAL_4]] : <i32>
// CHECK: }
// CHECK: } while {
// CHECK: %[[VAL_12:.*]] = expression %[[VAL_5]] : (!emitc.lvalue<i1>) -> i1 {
// CHECK: %[[VAL_13:.*]] = load %[[VAL_5]] : <i1>
// CHECK: yield %[[VAL_13]] : i1
// CHECK: }
// CHECK: yield %[[VAL_12]] : i1
// CHECK: }
// CHECK: %[[VAL_14:.*]] = emitc.load %[[VAL_3]] : <i32>
// CHECK: return %[[VAL_14]] : i32
// CHECK: }
emitc.func @payload_empty_after_region() -> i1 {
%true = emitc.literal "true" : i1
return %true : i1
}
func.func @empty_after_region() {
scf.while () : () -> () {
%condition = emitc.call @payload_empty_after_region() : () -> i1
scf.condition(%condition)
} do {
^bb0():
scf.yield
}
return
}
// CHECK-LABEL: emitc.func @payload_empty_after_region() -> i1 {
// CHECK: %[[VAL_0:.*]] = literal "true" : i1
// CHECK: return %[[VAL_0]] : i1
// CHECK: }
// CHECK-LABEL: func.func @empty_after_region() {
// CHECK: %[[VAL_0:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i1>
// CHECK: emitc.do {
// CHECK: %[[VAL_1:.*]] = call @payload_empty_after_region() : () -> i1
// CHECK: assign %[[VAL_1]] : i1 to %[[VAL_0]] : <i1>
// CHECK: } while {
// CHECK: %[[VAL_2:.*]] = expression %[[VAL_0]] : (!emitc.lvalue<i1>) -> i1 {
// CHECK: %[[VAL_3:.*]] = load %[[VAL_0]] : <i1>
// CHECK: yield %[[VAL_3]] : i1
// CHECK: }
// CHECK: yield %[[VAL_2]] : i1
// CHECK: }
// CHECK: return
// CHECK: }
emitc.func @payload_different_number_of_vars(%arg0: i32) -> i32 {
%0 = add %arg0, %arg0 : (i32, i32) -> i32
return %0 : i32
}
func.func @different_number_of_vars() -> (i32, i32) {
%init = emitc.literal "1.0" : i32
%var = emitc.literal "7.0" : i32
%exit = emitc.literal "10.0" : i32
%res, %res2 = scf.while (%arg1 = %init) : (i32) -> (i32, i32) {
%sum = emitc.add %arg1, %var : (i32, i32) -> i32
%condition = emitc.cmp lt, %sum, %exit : (i32, i32) -> i1
%next = emitc.add %arg1, %arg1 : (i32, i32) -> i32
scf.condition(%condition) %next, %sum : i32, i32
} do {
^bb0(%arg2: i32, %arg3 : i32):
%next_arg1 = emitc.call @payload_different_number_of_vars(%arg2) : (i32) -> i32
scf.yield %next_arg1 : i32
}
return %res, %res2 : i32, i32
}
// CHECK-LABEL: emitc.func @payload_different_number_of_vars(
// CHECK-SAME: %[[ARG0:.*]]: i32) -> i32 {
// CHECK: %[[VAL_0:.*]] = add %[[ARG0]], %[[ARG0]] : (i32, i32) -> i32
// CHECK: return %[[VAL_0]] : i32
// CHECK: }
// CHECK-LABEL: func.func @different_number_of_vars() -> (i32, i32) {
// CHECK: %[[VAL_0:.*]] = emitc.literal "1.0" : i32
// CHECK: %[[VAL_1:.*]] = emitc.literal "7.0" : i32
// CHECK: %[[VAL_2:.*]] = emitc.literal "10.0" : i32
// CHECK: %[[VAL_3:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i32>
// CHECK: %[[VAL_4:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i32>
// CHECK: %[[VAL_5:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i32>
// CHECK: emitc.assign %[[VAL_0]] : i32 to %[[VAL_5]] : <i32>
// CHECK: %[[VAL_6:.*]] = "emitc.variable"() <{value = #emitc.opaque<"">}> : () -> !emitc.lvalue<i1>
// CHECK: emitc.do {
// CHECK: %[[VAL_7:.*]] = load %[[VAL_5]] : <i32>
// CHECK: %[[VAL_8:.*]] = add %[[VAL_7]], %[[VAL_1]] : (i32, i32) -> i32
// CHECK: %[[VAL_9:.*]] = cmp lt, %[[VAL_8]], %[[VAL_2]] : (i32, i32) -> i1
// CHECK: %[[VAL_10:.*]] = add %[[VAL_7]], %[[VAL_7]] : (i32, i32) -> i32
// CHECK: assign %[[VAL_10]] : i32 to %[[VAL_3]] : <i32>
// CHECK: assign %[[VAL_8]] : i32 to %[[VAL_4]] : <i32>
// CHECK: assign %[[VAL_9]] : i1 to %[[VAL_6]] : <i1>
// CHECK: if %[[VAL_9]] {
// CHECK: %[[VAL_11:.*]] = call @payload_different_number_of_vars(%[[VAL_10]]) : (i32) -> i32
// CHECK: assign %[[VAL_11]] : i32 to %[[VAL_5]] : <i32>
// CHECK: }
// CHECK: } while {
// CHECK: %[[VAL_12:.*]] = expression %[[VAL_6]] : (!emitc.lvalue<i1>) -> i1 {
// CHECK: %[[VAL_13:.*]] = load %[[VAL_6]] : <i1>
// CHECK: yield %[[VAL_13]] : i1
// CHECK: }
// CHECK: yield %[[VAL_12]] : i1
// CHECK: }
// CHECK: %[[VAL_14:.*]] = emitc.load %[[VAL_3]] : <i32>
// CHECK: %[[VAL_15:.*]] = emitc.load %[[VAL_4]] : <i32>
// CHECK: return %[[VAL_14]], %[[VAL_15]] : i32, i32
// CHECK: }
|