blob: dfbb3f20ddbd8f2e716cb3dfd377bd400406f1d2 [file] [log] [blame]
Ethan Nicholas95046142021-01-07 10:57:27 -05001/*
2 * Copyright 2020 Google LLC
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
Ethan Nicholasdaed2592021-03-04 14:30:25 -05008#include "include/sksl/DSL.h"
Ethan Nicholas95046142021-01-07 10:57:27 -05009#include "src/gpu/GrDirectContextPriv.h"
10#include "src/gpu/GrGpu.h"
11#include "src/sksl/SkSLIRGenerator.h"
Ethan Nicholas95046142021-01-07 10:57:27 -050012#include "src/sksl/dsl/priv/DSLWriter.h"
Ethan Nicholas1ff76092021-01-28 10:02:43 -050013#include "src/sksl/ir/SkSLIRNode.h"
Ethan Nicholas95046142021-01-07 10:57:27 -050014
15#include "tests/Test.h"
16
Ethan Nicholasb3d4e742021-01-08 11:42:25 -050017#include <limits>
18
Ethan Nicholas95046142021-01-07 10:57:27 -050019using namespace SkSL::dsl;
20
21class AutoDSLContext {
22public:
23 AutoDSLContext(GrGpu* gpu) {
24 Start(gpu->shaderCompiler());
Ethan Nicholasbffe80a2021-01-11 15:42:44 -050025 DSLWriter::Instance().fMangle = false;
Ethan Nicholas95046142021-01-07 10:57:27 -050026 }
27
28 ~AutoDSLContext() {
29 End();
30 }
31};
32
Ethan Nicholasb3d4e742021-01-08 11:42:25 -050033class ExpectError : public ErrorHandler {
34public:
35 ExpectError(skiatest::Reporter* reporter, const char* msg)
36 : fMsg(msg)
37 , fReporter(reporter) {
38 SetErrorHandler(this);
39 }
40
41 ~ExpectError() override {
John Stiles642cde22021-02-23 14:57:01 -050042 REPORTER_ASSERT(fReporter, !fMsg,
43 "Error mismatch: expected:\n%sbut no error occurred\n", fMsg);
Ethan Nicholasb3d4e742021-01-08 11:42:25 -050044 SetErrorHandler(nullptr);
45 }
46
Ethan Nicholasb9563042021-02-25 09:45:49 -050047 void handleError(const char* msg, PositionInfo* pos) override {
Ethan Nicholasb3d4e742021-01-08 11:42:25 -050048 REPORTER_ASSERT(fReporter, !strcmp(msg, fMsg),
49 "Error mismatch: expected:\n%sbut received:\n%s", fMsg, msg);
50 fMsg = nullptr;
51 }
52
53private:
54 const char* fMsg;
55 skiatest::Reporter* fReporter;
56};
57
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -050058static bool whitespace_insensitive_compare(const char* a, const char* b) {
59 for (;;) {
60 while (isspace(*a)) {
61 ++a;
62 }
63 while (isspace(*b)) {
64 ++b;
65 }
66 if (*a != *b) {
67 return false;
68 }
69 if (*a == 0) {
70 return true;
71 }
72 ++a;
73 ++b;
74 }
75}
76
Ethan Nicholasb3d4e742021-01-08 11:42:25 -050077DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLStartup, r, ctxInfo) {
Ethan Nicholas95046142021-01-07 10:57:27 -050078 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
79 Expression e1 = 1;
80 REPORTER_ASSERT(r, e1.release()->description() == "1");
81 Expression e2 = 1.0;
82 REPORTER_ASSERT(r, e2.release()->description() == "1.0");
83 Expression e3 = true;
84 REPORTER_ASSERT(r, e3.release()->description() == "true");
Ethan Nicholasbffe80a2021-01-11 15:42:44 -050085 Var a(kInt, "a");
86 Expression e4 = a;
87 REPORTER_ASSERT(r, e4.release()->description() == "a");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -050088
89 REPORTER_ASSERT(r, whitespace_insensitive_compare("", ""));
90 REPORTER_ASSERT(r, !whitespace_insensitive_compare("", "a"));
91 REPORTER_ASSERT(r, !whitespace_insensitive_compare("a", ""));
92 REPORTER_ASSERT(r, whitespace_insensitive_compare("a", "a"));
93 REPORTER_ASSERT(r, whitespace_insensitive_compare("abc", "abc"));
94 REPORTER_ASSERT(r, whitespace_insensitive_compare("abc", " abc "));
95 REPORTER_ASSERT(r, whitespace_insensitive_compare("a b c ", "\n\n\nabc"));
96 REPORTER_ASSERT(r, !whitespace_insensitive_compare("a b c d", "\n\n\nabc"));
Ethan Nicholas95046142021-01-07 10:57:27 -050097}
Ethan Nicholasb3d4e742021-01-08 11:42:25 -050098
Ethan Nicholas34c7e112021-02-25 20:50:32 -050099static SkSL::String stringize(DSLStatement& stmt) { return stmt.release()->description(); }
100static SkSL::String stringize(DSLPossibleStatement& stmt) { return stmt.release()->description(); }
101static SkSL::String stringize(DSLExpression& expr) { return expr.release()->description(); }
102static SkSL::String stringize(DSLPossibleExpression& expr) { return expr.release()->description(); }
John Stilesb4d7b582021-02-19 09:56:31 -0500103static SkSL::String stringize(SkSL::IRNode& node) { return node.description(); }
104
105template <typename T>
106static void expect_equal(skiatest::Reporter* r, int lineNumber, T& input, const char* expected) {
107 SkSL::String actual = stringize(input);
108 if (!whitespace_insensitive_compare(expected, actual.c_str())) {
109 ERRORF(r, "(Failed on line %d)\nExpected: %s\n Actual: %s\n",
110 lineNumber, expected, actual.c_str());
111 }
112}
113
114template <typename T>
115static void expect_equal(skiatest::Reporter* r, int lineNumber, T&& dsl, const char* expected) {
116 // This overload allows temporary values to be passed to expect_equal.
117 return expect_equal(r, lineNumber, dsl, expected);
118}
119
120#define EXPECT_EQUAL(a, b) expect_equal(r, __LINE__, (a), (b))
121
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500122DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLFloat, r, ctxInfo) {
123 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
124 Expression e1 = Float(std::numeric_limits<float>::max());
125 REPORTER_ASSERT(r, atof(e1.release()->description().c_str()) ==
126 std::numeric_limits<float>::max());
127
128 Expression e2 = Float(std::numeric_limits<float>::min());
129 REPORTER_ASSERT(r, atof(e2.release()->description().c_str()) ==
130 std::numeric_limits<float>::min());
131
John Stilesb4d7b582021-02-19 09:56:31 -0500132 EXPECT_EQUAL(Float2(0),
133 "float2(0.0)");
134 EXPECT_EQUAL(Float2(-0.5, 1),
135 "float2(-0.5, 1.0)");
136 EXPECT_EQUAL(Float3(0.75),
137 "float3(0.75)");
138 EXPECT_EQUAL(Float3(Float2(0, 1), -2),
John Stilesb9e4f642021-03-05 09:11:38 -0500139 "float3(0.0, 1.0, -2.0)");
John Stilesb4d7b582021-02-19 09:56:31 -0500140 EXPECT_EQUAL(Float3(0, 1, 2),
141 "float3(0.0, 1.0, 2.0)");
142 EXPECT_EQUAL(Float4(0),
143 "float4(0.0)");
144 EXPECT_EQUAL(Float4(Float2(0, 1), Float2(2, 3)),
John Stilesb9e4f642021-03-05 09:11:38 -0500145 "float4(0.0, 1.0, 2.0, 3.0)");
John Stilesb4d7b582021-02-19 09:56:31 -0500146 EXPECT_EQUAL(Float4(0, 1, Float2(2, 3)),
John Stilesb9e4f642021-03-05 09:11:38 -0500147 "float4(0.0, 1.0, 2.0, 3.0)");
John Stilesb4d7b582021-02-19 09:56:31 -0500148 EXPECT_EQUAL(Float4(0, 1, 2, 3),
149 "float4(0.0, 1.0, 2.0, 3.0)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500150
151 {
152 ExpectError error(r, "error: floating point value is infinite\n");
153 Float(std::numeric_limits<float>::infinity()).release();
154 }
155
156 {
157 ExpectError error(r, "error: floating point value is NaN\n");
158 Float(std::numeric_limits<float>::quiet_NaN()).release();
159 }
160
161 {
162 ExpectError error(r, "error: invalid arguments to 'float2' constructor (expected 2 scalars,"
163 " but found 4)\n");
164 Float2(Float4(1)).release();
165 }
166
167 {
168 ExpectError error(r, "error: invalid arguments to 'float4' constructor (expected 4 scalars,"
169 " but found 3)\n");
170 Float4(Float3(1)).release();
171 }
172}
173
174DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLHalf, r, ctxInfo) {
175 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
176 Expression e1 = Half(std::numeric_limits<float>::max());
John Stilesb4d7b582021-02-19 09:56:31 -0500177 REPORTER_ASSERT(r,
178 atof(e1.release()->description().c_str()) == std::numeric_limits<float>::max());
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500179
180 Expression e2 = Half(std::numeric_limits<float>::min());
John Stilesb4d7b582021-02-19 09:56:31 -0500181 REPORTER_ASSERT(r,
182 atof(e2.release()->description().c_str()) == std::numeric_limits<float>::min());
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500183
John Stilesb9e4f642021-03-05 09:11:38 -0500184 EXPECT_EQUAL(Half2(0),
185 "half2(0.0)");
186 EXPECT_EQUAL(Half2(-0.5, 1),
187 "half2(-0.5, 1.0)");
188 EXPECT_EQUAL(Half3(0.75),
189 "half3(0.75)");
190 EXPECT_EQUAL(Half3(Half2(0, 1), -2),
191 "half3(0.0, 1.0, -2.0)");
192 EXPECT_EQUAL(Half3(0, 1, 2),
193 "half3(0.0, 1.0, 2.0)");
194 EXPECT_EQUAL(Half4(0),
195 "half4(0.0)");
196 EXPECT_EQUAL(Half4(Half2(0, 1), Half2(2, 3)),
197 "half4(0.0, 1.0, 2.0, 3.0)");
198 EXPECT_EQUAL(Half4(0, 1, Half2(2, 3)),
199 "half4(0.0, 1.0, 2.0, 3.0)");
200 EXPECT_EQUAL(Half4(0, 1, 2, 3),
201 "half4(0.0, 1.0, 2.0, 3.0)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500202
203 {
204 ExpectError error(r, "error: floating point value is infinite\n");
205 Half(std::numeric_limits<float>::infinity()).release();
206 }
207
208 {
209 ExpectError error(r, "error: floating point value is NaN\n");
210 Half(std::numeric_limits<float>::quiet_NaN()).release();
211 }
212
213 {
214 ExpectError error(r, "error: invalid arguments to 'half2' constructor (expected 2 scalars,"
215 " but found 4)\n");
216 Half2(Half4(1)).release();
217 }
218
219 {
220 ExpectError error(r, "error: invalid arguments to 'half4' constructor (expected 4 scalars,"
221 " but found 3)\n");
222 Half4(Half3(1)).release();
223 }
224}
225
226DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLInt, r, ctxInfo) {
227 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500228
John Stilesb9e4f642021-03-05 09:11:38 -0500229 EXPECT_EQUAL(Int(std::numeric_limits<int32_t>::max()),
230 "2147483647");
231 EXPECT_EQUAL(Int2(std::numeric_limits<int32_t>::min()),
232 "int2(-2147483648)");
233 EXPECT_EQUAL(Int2(0, 1),
234 "int2(0, 1)");
235 EXPECT_EQUAL(Int3(0),
236 "int3(0)");
237 EXPECT_EQUAL(Int3(Int2(0, 1), -2),
238 "int3(0, 1, -2)");
239 EXPECT_EQUAL(Int3(0, 1, 2),
240 "int3(0, 1, 2)");
241 EXPECT_EQUAL(Int4(0),
242 "int4(0)");
243 EXPECT_EQUAL(Int4(Int2(0, 1), Int2(2, 3)),
244 "int4(0, 1, 2, 3)");
245 EXPECT_EQUAL(Int4(0, 1, Int2(2, 3)),
246 "int4(0, 1, 2, 3)");
247 EXPECT_EQUAL(Int4(0, 1, 2, 3),
248 "int4(0, 1, 2, 3)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500249
250 {
251 ExpectError error(r, "error: invalid arguments to 'int2' constructor (expected 2 scalars,"
252 " but found 4)\n");
253 Int2(Int4(1)).release();
254 }
255
256 {
257 ExpectError error(r, "error: invalid arguments to 'int4' constructor (expected 4 scalars,"
258 " but found 3)\n");
259 Int4(Int3(1)).release();
260 }
261}
262
263DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLShort, r, ctxInfo) {
264 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500265
John Stilesb9e4f642021-03-05 09:11:38 -0500266 EXPECT_EQUAL(Short(std::numeric_limits<int16_t>::max()),
267 "32767");
268 EXPECT_EQUAL(Short2(std::numeric_limits<int16_t>::min()),
269 "short2(-32768)");
270 EXPECT_EQUAL(Short2(0, 1),
271 "short2(0, 1)");
272 EXPECT_EQUAL(Short3(0),
273 "short3(0)");
274 EXPECT_EQUAL(Short3(Short2(0, 1), -2),
275 "short3(0, 1, -2)");
276 EXPECT_EQUAL(Short3(0, 1, 2),
277 "short3(0, 1, 2)");
278 EXPECT_EQUAL(Short4(0),
279 "short4(0)");
280 EXPECT_EQUAL(Short4(Short2(0, 1), Short2(2, 3)),
281 "short4(0, 1, 2, 3)");
282 EXPECT_EQUAL(Short4(0, 1, Short2(2, 3)),
283 "short4(0, 1, 2, 3)");
284 EXPECT_EQUAL(Short4(0, 1, 2, 3),
285 "short4(0, 1, 2, 3)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500286
287 {
288 ExpectError error(r, "error: invalid arguments to 'short2' constructor (expected 2 scalars,"
289 " but found 4)\n");
290 Short2(Short4(1)).release();
291 }
292
293 {
294 ExpectError error(r, "error: invalid arguments to 'short4' constructor (expected 4 scalars,"
295 " but found 3)\n");
296 Short4(Short3(1)).release();
297 }
298}
299
300DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBool, r, ctxInfo) {
301 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500302
John Stilesb9e4f642021-03-05 09:11:38 -0500303 EXPECT_EQUAL(Bool2(false),
304 "bool2(false)");
305 EXPECT_EQUAL(Bool2(false, true),
306 "bool2(false, true)");
307 EXPECT_EQUAL(Bool3(false),
308 "bool3(false)");
309 EXPECT_EQUAL(Bool3(Bool2(false, true), false),
310 "bool3(false, true, false)");
311 EXPECT_EQUAL(Bool3(false, true, false),
312 "bool3(false, true, false)");
313 EXPECT_EQUAL(Bool4(false),
314 "bool4(false)");
315 EXPECT_EQUAL(Bool4(Bool2(false, true), Bool2(false, true)),
316 "bool4(false, true, false, true)");
317 EXPECT_EQUAL(Bool4(false, true, Bool2(false, true)),
318 "bool4(false, true, false, true)");
319 EXPECT_EQUAL(Bool4(false, true, false, true),
320 "bool4(false, true, false, true)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500321
322 {
323 ExpectError error(r, "error: invalid arguments to 'bool2' constructor (expected 2 scalars,"
324 " but found 4)\n");
325 Bool2(Bool4(true)).release();
326 }
327
328 {
329 ExpectError error(r, "error: invalid arguments to 'bool4' constructor (expected 4 scalars,"
330 " but found 3)\n");
331 Bool4(Bool3(true)).release();
332 }
333}
Ethan Nicholas92969f22021-01-13 10:38:59 -0500334
335DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLPlus, r, ctxInfo) {
336 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
337 Var a(kFloat, "a"), b(kFloat, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500338
John Stiles8f440b42021-03-05 16:48:56 -0500339 EXPECT_EQUAL(a + b,
340 "(a + b)");
341 EXPECT_EQUAL(a + 1,
342 "(a + 1.0)");
343 EXPECT_EQUAL(0.5 + a + -99,
344 "((0.5 + a) + -99.0)");
345 EXPECT_EQUAL(a += b + 1,
346 "(a += (b + 1.0))");
Ethan Nicholasb14b6362021-03-08 17:07:58 -0500347 EXPECT_EQUAL(+a,
348 "a");
349 EXPECT_EQUAL(+(a + b),
350 "(a + b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500351
352 {
353 ExpectError error(r, "error: type mismatch: '+' cannot operate on 'bool2', 'float'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500354 DSLExpression((Bool2(true) + a)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500355 }
356
357 {
358 ExpectError error(r, "error: type mismatch: '+=' cannot operate on 'float', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500359 DSLExpression((a += Bool2(true))).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500360 }
361
362 {
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500363 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500364 DSLExpression((1.0 += a)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500365 }
Ethan Nicholasb14b6362021-03-08 17:07:58 -0500366
367 {
368 ExpectError error(r, "error: '+' cannot operate on 'bool'\n");
369 Var c(kBool);
370 DSLExpression(+c);
371 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500372}
373
374DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLMinus, r, ctxInfo) {
375 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
376 Var a(kInt, "a"), b(kInt, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500377
John Stiles8f440b42021-03-05 16:48:56 -0500378 EXPECT_EQUAL(a - b,
379 "(a - b)");
380 EXPECT_EQUAL(a - 1,
381 "(a - 1)");
382 EXPECT_EQUAL(2 - a - b,
383 "((2 - a) - b)");
384 EXPECT_EQUAL(a -= b + 1,
385 "(a -= (b + 1))");
Ethan Nicholasb14b6362021-03-08 17:07:58 -0500386 EXPECT_EQUAL(-a,
387 "-a");
388 EXPECT_EQUAL(-(a - b),
389 "-(a - b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500390
391 {
392 ExpectError error(r, "error: type mismatch: '-' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500393 DSLExpression(Bool2(true) - a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500394 }
395
396 {
397 ExpectError error(r, "error: type mismatch: '-=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500398 DSLExpression(a -= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500399 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500400
401 {
402 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500403 DSLExpression(1.0 -= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500404 }
Ethan Nicholasb14b6362021-03-08 17:07:58 -0500405
406 {
407 ExpectError error(r, "error: '-' cannot operate on 'bool'\n");
408 Var c(kBool);
409 DSLExpression(-c);
410 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500411}
412
413DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLMultiply, r, ctxInfo) {
414 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
415 Var a(kFloat, "a"), b(kFloat, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500416
John Stiles8f440b42021-03-05 16:48:56 -0500417 EXPECT_EQUAL(a * b,
418 "(a * b)");
419 EXPECT_EQUAL(a * 2,
420 "(a * 2.0)");
421 EXPECT_EQUAL(0.5 * a * -99,
422 "((0.5 * a) * -99.0)");
423 EXPECT_EQUAL(a *= b + 1,
424 "(a *= (b + 1.0))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500425
426 {
427 ExpectError error(r, "error: type mismatch: '*' cannot operate on 'bool2', 'float'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500428 DSLExpression(Bool2(true) * a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500429 }
430
431 {
432 ExpectError error(r, "error: type mismatch: '*=' cannot operate on 'float', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500433 DSLExpression(a *= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500434 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500435
436 {
437 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500438 DSLExpression(1.0 *= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500439 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500440}
441
442DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDivide, r, ctxInfo) {
443 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
444 Var a(kFloat, "a"), b(kFloat, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500445
John Stiles8f440b42021-03-05 16:48:56 -0500446 EXPECT_EQUAL(a / b,
447 "(a / b)");
448 EXPECT_EQUAL(a / 2,
449 "(a / 2.0)");
450 EXPECT_EQUAL(0.5 / a / -99,
451 "((0.5 / a) / -99.0)");
452 EXPECT_EQUAL(b / (a - 1),
453 "(b / (a - 1.0))");
454 EXPECT_EQUAL(a /= b + 1,
455 "(a /= (b + 1.0))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500456
457 {
458 ExpectError error(r, "error: type mismatch: '/' cannot operate on 'bool2', 'float'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500459 DSLExpression(Bool2(true) / a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500460 }
461
462 {
463 ExpectError error(r, "error: type mismatch: '/=' cannot operate on 'float', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500464 DSLExpression(a /= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500465 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500466
467 {
468 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500469 DSLExpression(1.0 /= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500470 }
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500471
472 {
473 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500474 DSLExpression(a /= 0).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500475 }
476
477 {
478 Var c(kFloat2, "c");
479 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500480 DSLExpression(c /= Float2(Float(0), 1)).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500481 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500482}
483
484DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLMod, r, ctxInfo) {
485 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
486 Var a(kInt, "a"), b(kInt, "b");
487 Expression e1 = a % b;
John Stilesb4d7b582021-02-19 09:56:31 -0500488 EXPECT_EQUAL(e1, "(a % b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500489
490 Expression e2 = a % 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500491 EXPECT_EQUAL(e2, "(a % 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500492
493 Expression e3 = 10 % a % -99;
John Stilesb4d7b582021-02-19 09:56:31 -0500494 EXPECT_EQUAL(e3, "((10 % a) % -99)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500495
496 Expression e4 = a %= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500497 EXPECT_EQUAL(e4, "(a %= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500498
499 {
500 ExpectError error(r, "error: type mismatch: '%' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500501 DSLExpression(Bool2(true) % a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500502 }
503
504 {
505 ExpectError error(r, "error: type mismatch: '%=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500506 DSLExpression(a %= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500507 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500508
509 {
510 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500511 DSLExpression(1 %= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500512 }
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500513
514 {
515 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500516 DSLExpression(a %= 0).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500517 }
518
519 {
520 Var c(kInt2, "c");
521 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500522 DSLExpression(c %= Int2(Int(0), 1)).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500523 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500524}
525
526DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLShl, r, ctxInfo) {
527 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
528 Var a(kInt, "a"), b(kInt, "b");
529 Expression e1 = a << b;
John Stilesb4d7b582021-02-19 09:56:31 -0500530 EXPECT_EQUAL(e1, "(a << b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500531
532 Expression e2 = a << 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500533 EXPECT_EQUAL(e2, "(a << 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500534
535 Expression e3 = 1 << a << 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500536 EXPECT_EQUAL(e3, "((1 << a) << 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500537
538 Expression e4 = a <<= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500539 EXPECT_EQUAL(e4, "(a <<= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500540
541 {
542 ExpectError error(r, "error: type mismatch: '<<' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500543 DSLExpression(Bool2(true) << a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500544 }
545
546 {
547 ExpectError error(r, "error: type mismatch: '<<=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500548 DSLExpression(a <<= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500549 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500550
551 {
552 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500553 DSLExpression(1 <<= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500554 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500555}
556
557DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLShr, r, ctxInfo) {
558 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
559 Var a(kInt, "a"), b(kInt, "b");
560 Expression e1 = a >> b;
John Stilesb4d7b582021-02-19 09:56:31 -0500561 EXPECT_EQUAL(e1, "(a >> b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500562
563 Expression e2 = a >> 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500564 EXPECT_EQUAL(e2, "(a >> 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500565
566 Expression e3 = 1 >> a >> 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500567 EXPECT_EQUAL(e3, "((1 >> a) >> 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500568
569 Expression e4 = a >>= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500570 EXPECT_EQUAL(e4, "(a >>= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500571
572 {
573 ExpectError error(r, "error: type mismatch: '>>' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500574 DSLExpression(Bool2(true) >> a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500575 }
576
577 {
578 ExpectError error(r, "error: type mismatch: '>>=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500579 DSLExpression(a >>= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500580 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500581
582 {
583 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500584 DSLExpression(1 >>= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500585 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500586}
587
588DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseAnd, r, ctxInfo) {
589 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
590 Var a(kInt, "a"), b(kInt, "b");
591 Expression e1 = a & b;
John Stilesb4d7b582021-02-19 09:56:31 -0500592 EXPECT_EQUAL(e1, "(a & b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500593
594 Expression e2 = a & 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500595 EXPECT_EQUAL(e2, "(a & 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500596
597 Expression e3 = 1 & a & 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500598 EXPECT_EQUAL(e3, "((1 & a) & 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500599
600 Expression e4 = a &= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500601 EXPECT_EQUAL(e4, "(a &= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500602
603 {
604 ExpectError error(r, "error: type mismatch: '&' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500605 DSLExpression(Bool2(true) & a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500606 }
607
608 {
609 ExpectError error(r, "error: type mismatch: '&=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500610 DSLExpression(a &= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500611 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500612
613 {
614 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500615 DSLExpression(1 &= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500616 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500617}
618
619DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseOr, r, ctxInfo) {
620 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
621 Var a(kInt, "a"), b(kInt, "b");
622 Expression e1 = a | b;
John Stilesb4d7b582021-02-19 09:56:31 -0500623 EXPECT_EQUAL(e1, "(a | b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500624
625 Expression e2 = a | 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500626 EXPECT_EQUAL(e2, "(a | 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500627
628 Expression e3 = 1 | a | 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500629 EXPECT_EQUAL(e3, "((1 | a) | 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500630
631 Expression e4 = a |= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500632 EXPECT_EQUAL(e4, "(a |= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500633
634 {
635 ExpectError error(r, "error: type mismatch: '|' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500636 DSLExpression(Bool2(true) | a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500637 }
638
639 {
640 ExpectError error(r, "error: type mismatch: '|=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500641 DSLExpression(a |= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500642 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500643
644 {
645 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500646 DSLExpression(1 |= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500647 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500648}
649
650DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseXor, r, ctxInfo) {
651 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
652 Var a(kInt, "a"), b(kInt, "b");
653 Expression e1 = a ^ b;
John Stilesb4d7b582021-02-19 09:56:31 -0500654 EXPECT_EQUAL(e1, "(a ^ b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500655
656 Expression e2 = a ^ 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500657 EXPECT_EQUAL(e2, "(a ^ 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500658
659 Expression e3 = 1 ^ a ^ 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500660 EXPECT_EQUAL(e3, "((1 ^ a) ^ 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500661
662 Expression e4 = a ^= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500663 EXPECT_EQUAL(e4, "(a ^= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500664
665 {
666 ExpectError error(r, "error: type mismatch: '^' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500667 DSLExpression(Bool2(true) ^ a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500668 }
669
670 {
671 ExpectError error(r, "error: type mismatch: '^=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500672 DSLExpression(a ^= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500673 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500674
675 {
676 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500677 DSLExpression(1 ^= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500678 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500679}
680
681DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLogicalAnd, r, ctxInfo) {
682 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
683 Var a(kBool, "a"), b(kBool, "b");
684 Expression e1 = a && b;
John Stilesb4d7b582021-02-19 09:56:31 -0500685 EXPECT_EQUAL(e1, "(a && b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500686
687 Expression e2 = a && true && b;
John Stilesb4d7b582021-02-19 09:56:31 -0500688 EXPECT_EQUAL(e2, "(a && b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500689
690 Expression e3 = a && false && b;
John Stilesb4d7b582021-02-19 09:56:31 -0500691 EXPECT_EQUAL(e3, "false");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500692
693 {
694 ExpectError error(r, "error: type mismatch: '&&' cannot operate on 'bool', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500695 DSLExpression(a && 5).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500696 }
697}
698
699DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLogicalOr, r, ctxInfo) {
700 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
701 Var a(kBool, "a"), b(kBool, "b");
702 Expression e1 = a || b;
John Stilesb4d7b582021-02-19 09:56:31 -0500703 EXPECT_EQUAL(e1, "(a || b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500704
705 Expression e2 = a || true || b;
John Stilesb4d7b582021-02-19 09:56:31 -0500706 EXPECT_EQUAL(e2, "true");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500707
708 Expression e3 = a || false || b;
John Stilesb4d7b582021-02-19 09:56:31 -0500709 EXPECT_EQUAL(e3, "(a || b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500710
711 {
712 ExpectError error(r, "error: type mismatch: '||' cannot operate on 'bool', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500713 DSLExpression(a || 5).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500714 }
715}
716
717DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLComma, r, ctxInfo) {
718 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
719 Var a(kInt, "a"), b(kInt, "b");
720 Expression e1 = (a += b, b);
John Stilesb4d7b582021-02-19 09:56:31 -0500721 EXPECT_EQUAL(e1, "((a += b) , b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500722
723 Expression e2 = (a += b, b += b, Int2(a));
John Stilesb4d7b582021-02-19 09:56:31 -0500724 EXPECT_EQUAL(e2, "(((a += b) , (b += b)) , int2(a))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500725}
726
727DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLEqual, r, ctxInfo) {
728 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
729 Var a(kInt, "a"), b(kInt, "b");
730 Expression e1 = a == b;
John Stilesb4d7b582021-02-19 09:56:31 -0500731 EXPECT_EQUAL(e1, "(a == b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500732
733 Expression e2 = a == 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500734 EXPECT_EQUAL(e2, "(a == 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500735
736 {
737 ExpectError error(r, "error: type mismatch: '==' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500738 DSLExpression(a == Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500739 }
740}
741
742DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLNotEqual, r, ctxInfo) {
743 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
744 Var a(kInt, "a"), b(kInt, "b");
745 Expression e1 = a != b;
John Stilesb4d7b582021-02-19 09:56:31 -0500746 EXPECT_EQUAL(e1, "(a != b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500747
748 Expression e2 = a != 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500749 EXPECT_EQUAL(e2, "(a != 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500750
751 {
752 ExpectError error(r, "error: type mismatch: '!=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500753 DSLExpression(a != Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500754 }
755}
756
757DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLGreaterThan, r, ctxInfo) {
758 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
759 Var a(kInt, "a"), b(kInt, "b");
760 Expression e1 = a > b;
John Stilesb4d7b582021-02-19 09:56:31 -0500761 EXPECT_EQUAL(e1, "(a > b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500762
763 Expression e2 = a > 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500764 EXPECT_EQUAL(e2, "(a > 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500765
766 {
767 ExpectError error(r, "error: type mismatch: '>' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500768 DSLExpression(a > Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500769 }
770}
771
772DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLGreaterThanOrEqual, r, ctxInfo) {
773 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
774 Var a(kInt, "a"), b(kInt, "b");
775 Expression e1 = a >= b;
John Stilesb4d7b582021-02-19 09:56:31 -0500776 EXPECT_EQUAL(e1, "(a >= b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500777
778 Expression e2 = a >= 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500779 EXPECT_EQUAL(e2, "(a >= 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500780
781 {
782 ExpectError error(r, "error: type mismatch: '>=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500783 DSLExpression(a >= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500784 }
785}
786
787DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLessThan, r, ctxInfo) {
788 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
789 Var a(kInt, "a"), b(kInt, "b");
790 Expression e1 = a < b;
John Stilesb4d7b582021-02-19 09:56:31 -0500791 EXPECT_EQUAL(e1, "(a < b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500792
793 Expression e2 = a < 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500794 EXPECT_EQUAL(e2, "(a < 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500795
796 {
797 ExpectError error(r, "error: type mismatch: '<' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500798 DSLExpression(a < Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500799 }
800}
801
802DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLessThanOrEqual, r, ctxInfo) {
803 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
804 Var a(kInt, "a"), b(kInt, "b");
805 Expression e1 = a <= b;
John Stilesb4d7b582021-02-19 09:56:31 -0500806 EXPECT_EQUAL(e1, "(a <= b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500807
808 Expression e2 = a <= 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500809 EXPECT_EQUAL(e2, "(a <= 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500810
811 {
812 ExpectError error(r, "error: type mismatch: '<=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500813 DSLExpression(a <= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500814 }
815}
816
817DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLogicalNot, r, ctxInfo) {
818 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
819 Var a(kInt, "a"), b(kInt, "b");
820 Expression e1 = !(a <= b);
John Stilesb4d7b582021-02-19 09:56:31 -0500821 EXPECT_EQUAL(e1, "!(a <= b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500822
823 {
824 ExpectError error(r, "error: '!' cannot operate on 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500825 DSLExpression(!a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500826 }
827}
828
829DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseNot, r, ctxInfo) {
830 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
831 Var a(kInt, "a"), b(kBool, "b");
832 Expression e1 = ~a;
John Stilesb4d7b582021-02-19 09:56:31 -0500833 EXPECT_EQUAL(e1, "~a");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500834
835 {
836 ExpectError error(r, "error: '~' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500837 DSLExpression(~b).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500838 }
839}
840
841DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLIncrement, r, ctxInfo) {
842 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
843 Var a(kInt, "a"), b(kBool, "b");
844 Expression e1 = ++a;
John Stilesb4d7b582021-02-19 09:56:31 -0500845 EXPECT_EQUAL(e1, "++a");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500846
847 Expression e2 = a++;
John Stilesb4d7b582021-02-19 09:56:31 -0500848 EXPECT_EQUAL(e2, "a++");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500849
850 {
851 ExpectError error(r, "error: '++' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500852 DSLExpression(++b).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500853 }
854
855 {
856 ExpectError error(r, "error: '++' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500857 DSLExpression(b++).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500858 }
859
860 {
861 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500862 DSLExpression(++(a + 1)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500863 }
864
865 {
866 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500867 DSLExpression((a + 1)++).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500868 }
869}
870
871DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDecrement, r, ctxInfo) {
872 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
873 Var a(kInt, "a"), b(kBool, "b");
874 Expression e1 = --a;
John Stilesb4d7b582021-02-19 09:56:31 -0500875 EXPECT_EQUAL(e1, "--a");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500876
877 Expression e2 = a--;
John Stilesb4d7b582021-02-19 09:56:31 -0500878 EXPECT_EQUAL(e2, "a--");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500879
880 {
881 ExpectError error(r, "error: '--' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500882 DSLExpression(--b).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500883 }
884
885 {
886 ExpectError error(r, "error: '--' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500887 DSLExpression(b--).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500888 }
889
890 {
891 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500892 DSLExpression(--(a + 1)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500893 }
894
895 {
896 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500897 DSLExpression((a + 1)--).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500898 }
899}
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500900
901DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBlock, r, ctxInfo) {
902 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
903 Statement x = Block();
John Stilesb4d7b582021-02-19 09:56:31 -0500904 EXPECT_EQUAL(x, "{ }");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500905 Var a(kInt, "a", 1), b(kInt, "b", 2);
906 Statement y = Block(Declare(a), Declare(b), a = b);
John Stilesb4d7b582021-02-19 09:56:31 -0500907 EXPECT_EQUAL(y, "{ int a = 1; int b = 2; (a = b); }");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500908}
909
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500910DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBreak, r, ctxInfo) {
911 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500912 Var i(kInt, "i", 0);
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500913 DSLFunction(kVoid, "success").define(
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500914 For(Declare(i), i < 10, ++i, Block(
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500915 If(i > 5, Break())
916 ))
917 );
918 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
John Stilesb4d7b582021-02-19 09:56:31 -0500919 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
920 "void success() { for (int i = 0; (i < 10); ++i) { if ((i > 5)) break; } }");
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500921
922 {
923 ExpectError error(r, "error: break statement must be inside a loop or switch\n");
924 DSLFunction(kVoid, "fail").define(
925 Break()
926 );
927 }
928}
929
930DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLContinue, r, ctxInfo) {
931 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500932 Var i(kInt, "i", 0);
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500933 DSLFunction(kVoid, "success").define(
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500934 For(Declare(i), i < 10, ++i, Block(
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500935 If(i < 5, Continue())
936 ))
937 );
938 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
John Stilesb4d7b582021-02-19 09:56:31 -0500939 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
940 "void success() { for (int i = 0; (i < 10); ++i) { if ((i < 5)) continue; } }");
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500941
942 {
943 ExpectError error(r, "error: continue statement must be inside a loop\n");
944 DSLFunction(kVoid, "fail").define(
945 Continue()
946 );
947 }
948}
949
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500950DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDeclare, r, ctxInfo) {
951 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500952 Var a(kHalf4, "a"), b(kHalf4, "b", Half4(1));
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500953 Statement x = Declare(a);
John Stilesb4d7b582021-02-19 09:56:31 -0500954 EXPECT_EQUAL(x, "half4 a;");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500955 Statement y = Declare(b);
John Stilesb4d7b582021-02-19 09:56:31 -0500956 EXPECT_EQUAL(y, "half4 b = half4(1.0);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500957
958 {
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500959 Var c(kHalf4, "c", 1);
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500960 ExpectError error(r, "error: expected 'half4', but found 'int'\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500961 Declare(c).release();
962 }
963
964 {
965 Var d(kInt, "d");
966 Declare(d).release();
967 ExpectError error(r, "error: variable has already been declared\n");
968 Declare(d).release();
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500969 }
970}
971
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500972DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDiscard, r, ctxInfo) {
973 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
974 Statement x = If(Sqrt(1) > 0, Discard());
John Stilesb4d7b582021-02-19 09:56:31 -0500975 EXPECT_EQUAL(x, "if ((sqrt(1.0) > 0.0)) discard;");
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500976}
977
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500978DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDo, r, ctxInfo) {
979 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
980 Statement x = Do(Block(), true);
John Stilesb4d7b582021-02-19 09:56:31 -0500981 EXPECT_EQUAL(x, "do {} while (true);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500982
983 Var a(kFloat, "a"), b(kFloat, "b");
984 Statement y = Do(Block(a++, --b), a != b);
John Stilesb4d7b582021-02-19 09:56:31 -0500985 EXPECT_EQUAL(y, "do { a++; --b; } while ((a != b));");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500986
987 {
988 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
989 Do(Block(), 7).release();
990 }
991}
992
993DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLFor, r, ctxInfo) {
994 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
995 Statement x = For(Statement(), Expression(), Expression(), Block());
John Stilesb4d7b582021-02-19 09:56:31 -0500996 EXPECT_EQUAL(x, "for (;;) {}");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500997
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500998 Var i(kInt, "i", 0);
999 Statement y = For(Declare(i), i < 10, ++i, i += 5);
John Stilesb4d7b582021-02-19 09:56:31 -05001000 EXPECT_EQUAL(y, "for (int i = 0; (i < 10); ++i) (i += 5);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001001
1002 {
1003 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
1004 For(i = 0, i + 10, ++i, i += 5).release();
1005 }
1006}
1007
Ethan Nicholase2c05042021-02-03 10:27:22 -05001008DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLFunction, r, ctxInfo) {
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001009 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001010 Var coords(kHalf2, "coords");
1011 DSLFunction(kVoid, "main", coords).define(
1012 sk_FragColor() = Half4(coords, 0, 1)
1013 );
1014 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
John Stilesb4d7b582021-02-19 09:56:31 -05001015 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1016 "void main(half2 coords) { (sk_FragColor = half4(coords, 0.0, 1.0)); }");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001017
Ethan Nicholas63f75fc2021-02-23 12:05:49 -05001018 {
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001019 DSLWriter::Reset();
Ethan Nicholas63f75fc2021-02-23 12:05:49 -05001020 Var x(kFloat, "x");
1021 DSLFunction sqr(kFloat, "sqr", x);
1022 sqr.define(
1023 Return(x * x)
1024 );
1025 EXPECT_EQUAL(sqr(sk_FragCoord().x()), "sqr(sk_FragCoord.x)");
1026 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
1027 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0], "float sqr(float x) { return (x * x); }");
1028 }
1029
1030 {
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001031 DSLWriter::Reset();
Ethan Nicholas63f75fc2021-02-23 12:05:49 -05001032 Var x(kFloat2, "x");
1033 Var y(kFloat2, "y");
1034 DSLFunction dot(kFloat2, "dot", x, y);
1035 dot.define(
1036 Return(x * x + y * y)
1037 );
1038 EXPECT_EQUAL(dot(Float2(1.0f, 2.0f), Float2(3.0f, 4.0f)),
1039 "dot(float2(1.0, 2.0), float2(3.0, 4.0))");
1040 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
1041 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1042 "float2 dot(float2 x, float2 y) { return ((x * x) + (y * y)); }");
1043 }
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001044
1045 {
1046 ExpectError error(r, "error: expected 'float', but found 'bool'\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001047 DSLWriter::Reset();
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001048 DSLFunction(kFloat, "broken").define(
1049 Return(true)
1050 );
1051 }
1052
1053 {
John Stilesb3dcbb12021-03-04 16:00:20 -05001054 ExpectError error(r, "error: expected function to return 'float'\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001055 DSLWriter::Reset();
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001056 DSLFunction(kFloat, "broken").define(
1057 Return()
1058 );
1059 }
1060
1061 {
John Stilesb3dcbb12021-03-04 16:00:20 -05001062 ExpectError error(r, "error: function 'broken' can exit without returning a value\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001063 DSLWriter::Reset();
1064 Var x(kFloat, "x", 0);
John Stilesb3dcbb12021-03-04 16:00:20 -05001065 DSLFunction(kFloat, "broken").define(
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001066 Declare(x),
John Stilesb3dcbb12021-03-04 16:00:20 -05001067 If(x == 1, Return(x))
1068 );
1069 }
1070
1071 {
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001072 ExpectError error(r, "error: may not return a value from a void function\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001073 DSLWriter::Reset();
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001074 DSLFunction(kVoid, "broken").define(
1075 Return(0)
1076 );
1077 }
1078
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001079 {
John Stilesb3dcbb12021-03-04 16:00:20 -05001080 ExpectError error(r, "error: function 'broken' can exit without returning a value\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001081 DSLWriter::Reset();
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001082 DSLFunction(kFloat, "broken").define(
1083 );
1084 }
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001085}
1086
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001087DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLIf, r, ctxInfo) {
1088 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1089 Var a(kFloat, "a"), b(kFloat, "b");
1090 Statement x = If(a > b, a -= b);
John Stilesb4d7b582021-02-19 09:56:31 -05001091 EXPECT_EQUAL(x, "if ((a > b)) (a -= b);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001092
1093 Statement y = If(a > b, a -= b, b -= a);
John Stilesb4d7b582021-02-19 09:56:31 -05001094 EXPECT_EQUAL(y, "if ((a > b)) (a -= b); else (b -= a);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001095
1096 {
1097 ExpectError error(r, "error: expected 'bool', but found 'float'\n");
1098 If(a + b, a -= b).release();
1099 }
1100}
1101
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001102DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLReturn, r, ctxInfo) {
1103 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1104
1105 Statement x = Return();
John Stilesb4d7b582021-02-19 09:56:31 -05001106 EXPECT_EQUAL(x, "return;");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001107
1108 Statement y = Return(true);
John Stilesb4d7b582021-02-19 09:56:31 -05001109 EXPECT_EQUAL(y, "return true;");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001110}
1111
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001112DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSelect, r, ctxInfo) {
1113 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1114 Var a(kInt, "a");
1115 Expression x = Select(a > 0, 1, -1);
John Stilesb4d7b582021-02-19 09:56:31 -05001116 EXPECT_EQUAL(x, "((a > 0) ? 1 : -1)");
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001117
1118 {
1119 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001120 DSLExpression x = Select(a, 1, -1);
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001121 }
1122
1123 {
1124 ExpectError error(r, "error: ternary operator result mismatch: 'float2', 'float3'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001125 DSLExpression x = Select(a > 0, Float2(1), Float3(1));
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001126 }
1127}
1128
Ethan Nicholascfefec02021-02-09 15:22:57 -05001129DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSwitch, r, ctxInfo) {
1130 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1131
1132 Var a(kFloat, "a"), b(kInt, "b");
1133
1134 Statement x = Switch(5,
1135 Case(0, a = 0, Break()),
1136 Case(1, a = 1, Continue()),
John Stilese1d1b082021-02-23 13:44:36 -05001137 Case(2, a = 2 /*Fallthrough*/),
Ethan Nicholascfefec02021-02-09 15:22:57 -05001138 Default(Discard())
1139 );
John Stilese1d1b082021-02-23 13:44:36 -05001140 EXPECT_EQUAL(x, R"(
1141 switch (5) {
1142 case 0: (a = 0.0); break;
1143 case 1: (a = 1.0); continue;
1144 case 2: (a = 2.0);
1145 default: discard;
1146 }
1147 )");
Ethan Nicholascfefec02021-02-09 15:22:57 -05001148
John Stiles642cde22021-02-23 14:57:01 -05001149 EXPECT_EQUAL(Switch(b),
1150 "switch (b) {}");
1151
1152 EXPECT_EQUAL(Switch(b, Default(), Case(0), Case(1)),
1153 "switch (b) { default: case 0: case 1: }");
Ethan Nicholascfefec02021-02-09 15:22:57 -05001154
1155 {
John Stilese1d1b082021-02-23 13:44:36 -05001156 ExpectError error(r, "error: duplicate case value '0'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001157 DSLStatement(Switch(0, Case(0), Case(0))).release();
Ethan Nicholascfefec02021-02-09 15:22:57 -05001158 }
1159
1160 {
John Stilese1d1b082021-02-23 13:44:36 -05001161 ExpectError error(r, "error: duplicate default case\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001162 DSLStatement(Switch(0, Default(a = 0), Default(a = 1))).release();
John Stilese1d1b082021-02-23 13:44:36 -05001163 }
1164
1165 {
Ethan Nicholascfefec02021-02-09 15:22:57 -05001166 ExpectError error(r, "error: case value must be a constant integer\n");
1167 Var b(kInt);
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001168 DSLStatement(Switch(0, Case(b))).release();
Ethan Nicholascfefec02021-02-09 15:22:57 -05001169 }
1170
1171 {
1172 ExpectError error(r, "error: continue statement must be inside a loop\n");
1173 DSLFunction(kVoid, "fail").define(
1174 Switch(5,
1175 Case(0, a = 0, Break()),
1176 Case(1, a = 1, Continue()),
1177 Default(Discard())
1178 )
1179 );
1180 }
1181}
1182
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001183DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSwizzle, r, ctxInfo) {
1184 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1185 Var a(kFloat4, "a");
1186
John Stilesf04e09c2021-03-05 13:13:14 -05001187 EXPECT_EQUAL(a.x(),
1188 "a.x");
1189 EXPECT_EQUAL(a.y(),
1190 "a.y");
1191 EXPECT_EQUAL(a.z(),
1192 "a.z");
1193 EXPECT_EQUAL(a.w(),
1194 "a.w");
1195 EXPECT_EQUAL(a.r(),
1196 "a.x");
1197 EXPECT_EQUAL(a.g(),
1198 "a.y");
1199 EXPECT_EQUAL(a.b(),
1200 "a.z");
1201 EXPECT_EQUAL(a.a(),
1202 "a.w");
1203 EXPECT_EQUAL(Swizzle(a, R),
1204 "a.x");
1205 EXPECT_EQUAL(Swizzle(a, ZERO, G),
1206 "float2(0.0, a.y)");
1207 EXPECT_EQUAL(Swizzle(a, B, G, G),
1208 "a.zyy");
1209 EXPECT_EQUAL(Swizzle(a, R, G, B, ONE),
1210 "float4(a.xyz, 1.0)");
1211 EXPECT_EQUAL(Swizzle(a, B, G, R, ONE).r(),
1212 "a.z");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001213}
1214
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001215DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLWhile, r, ctxInfo) {
1216 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1217 Statement x = While(true, Block());
John Stilesb4d7b582021-02-19 09:56:31 -05001218 EXPECT_EQUAL(x, "for (; true;) {}");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001219
1220 Var a(kFloat, "a"), b(kFloat, "b");
1221 Statement y = While(a != b, Block(a++, --b));
John Stilesb4d7b582021-02-19 09:56:31 -05001222 EXPECT_EQUAL(y, "for (; (a != b);) { a++; --b; }");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001223
1224 {
1225 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001226 DSLStatement x = While(7, Block());
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001227 }
1228}
Ethan Nicholas04be3392021-01-26 10:07:01 -05001229
1230DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLIndex, r, ctxInfo) {
1231 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1232 Var a(Array(kInt, 5), "a"), b(kInt, "b");
John Stilesb4d7b582021-02-19 09:56:31 -05001233
1234 EXPECT_EQUAL(a[0], "a[0]");
1235 EXPECT_EQUAL(a[b], "a[b]");
Ethan Nicholas04be3392021-01-26 10:07:01 -05001236
1237 {
1238 ExpectError error(r, "error: expected 'int', but found 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001239 DSLExpression x = a[true];
Ethan Nicholas04be3392021-01-26 10:07:01 -05001240 }
1241
1242 {
1243 ExpectError error(r, "error: expected array, but found 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001244 DSLExpression x = b[0];
Ethan Nicholas04be3392021-01-26 10:07:01 -05001245 }
1246
1247 {
1248 ExpectError error(r, "error: index -1 out of range for 'int[5]'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001249 DSLExpression x = a[-1];
Ethan Nicholas04be3392021-01-26 10:07:01 -05001250 }
1251}
Ethan Nicholas30e93d52021-01-26 12:00:25 -05001252
1253DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBuiltins, r, ctxInfo) {
1254 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1255 // There is a Fract type on Mac which can conflict with our Fract builtin
1256 using SkSL::dsl::Fract;
1257 Var a(kHalf4, "a"), b(kHalf4, "b"), c(kHalf4, "c");
1258 Var h3(kHalf3, "h3");
1259 Var b4(kBool4, "b4");
John Stilesb4d7b582021-02-19 09:56:31 -05001260 EXPECT_EQUAL(Abs(a), "abs(a)");
1261 EXPECT_EQUAL(All(b4), "all(b4)");
1262 EXPECT_EQUAL(Any(b4), "any(b4)");
1263 EXPECT_EQUAL(Ceil(a), "ceil(a)");
1264 EXPECT_EQUAL(Clamp(a, 0, 1), "clamp(a, 0.0, 1.0)");
1265 EXPECT_EQUAL(Cos(a), "cos(a)");
1266 EXPECT_EQUAL(Cross(h3, h3), "cross(h3, h3)");
1267 EXPECT_EQUAL(Degrees(a), "degrees(a)");
1268 EXPECT_EQUAL(Distance(a, b), "distance(a, b)");
1269 EXPECT_EQUAL(Dot(a, b), "dot(a, b)");
1270 EXPECT_EQUAL(Equal(a, b), "equal(a, b)");
1271 EXPECT_EQUAL(Exp(a), "exp(a)");
1272 EXPECT_EQUAL(Exp2(a), "exp2(a)");
1273 EXPECT_EQUAL(Faceforward(a, b, c), "faceforward(a, b, c)");
1274 EXPECT_EQUAL(Floor(a), "floor(a)");
1275 EXPECT_EQUAL(Fract(a), "fract(a)");
1276 EXPECT_EQUAL(GreaterThan(a, b), "greaterThan(a, b)");
1277 EXPECT_EQUAL(GreaterThanEqual(a, b), "greaterThanEqual(a, b)");
1278 EXPECT_EQUAL(Inversesqrt(a), "inversesqrt(a)");
1279 EXPECT_EQUAL(LessThan(a, b), "lessThan(a, b)");
1280 EXPECT_EQUAL(LessThanEqual(a, b), "lessThanEqual(a, b)");
1281 EXPECT_EQUAL(Length(a), "length(a)");
1282 EXPECT_EQUAL(Log(a), "log(a)");
1283 EXPECT_EQUAL(Log2(a), "log2(a)");
1284 EXPECT_EQUAL(Max(a, b), "max(a, b)");
1285 EXPECT_EQUAL(Min(a, b), "min(a, b)");
1286 EXPECT_EQUAL(Mix(a, b, c), "mix(a, b, c)");
1287 EXPECT_EQUAL(Mod(a, b), "mod(a, b)");
1288 EXPECT_EQUAL(Normalize(a), "normalize(a)");
1289 EXPECT_EQUAL(NotEqual(a, b), "notEqual(a, b)");
1290 EXPECT_EQUAL(Pow(a, b), "pow(a, b)");
1291 EXPECT_EQUAL(Radians(a), "radians(a)");
1292 EXPECT_EQUAL(Reflect(a, b), "reflect(a, b)");
1293 EXPECT_EQUAL(Refract(a, b, 1), "refract(a, b, 1.0)");
1294 EXPECT_EQUAL(Saturate(a), "saturate(a)");
1295 EXPECT_EQUAL(Sign(a), "sign(a)");
1296 EXPECT_EQUAL(Sin(a), "sin(a)");
1297 EXPECT_EQUAL(Smoothstep(a, b, c), "smoothstep(a, b, c)");
1298 EXPECT_EQUAL(Sqrt(a), "sqrt(a)");
1299 EXPECT_EQUAL(Step(a, b), "step(a, b)");
1300 EXPECT_EQUAL(Tan(a), "tan(a)");
1301 EXPECT_EQUAL(Unpremul(a), "unpremul(a)");
Ethan Nicholas30e93d52021-01-26 12:00:25 -05001302
1303 // these calls all go through the normal channels, so it ought to be sufficient to prove that
1304 // one of them reports errors correctly
1305 {
1306 ExpectError error(r, "error: no match for ceil(bool)\n");
1307 Ceil(a == b).release();
1308 }
1309}
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001310
1311DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLModifiers, r, ctxInfo) {
1312 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1313
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001314 Var v1(kConst_Modifier, kInt, "v1", 0);
1315 Statement d1 = Declare(v1);
Ethan Nicholasbd974002021-02-22 16:20:06 -05001316 EXPECT_EQUAL(d1, "const int v1 = 0;");
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001317
1318 // Most modifiers require an appropriate context to be legal. We can't yet give them that
1319 // context, so we can't as yet Declare() variables with these modifiers.
1320 // TODO: better tests when able
1321 Var v2(kIn_Modifier, kInt, "v2");
1322 REPORTER_ASSERT(r, DSLWriter::Var(v2).modifiers().fFlags == SkSL::Modifiers::kIn_Flag);
1323
1324 Var v3(kOut_Modifier, kInt, "v3");
1325 REPORTER_ASSERT(r, DSLWriter::Var(v3).modifiers().fFlags == SkSL::Modifiers::kOut_Flag);
1326
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001327 Var v4(kFlat_Modifier, kInt, "v4");
1328 REPORTER_ASSERT(r, DSLWriter::Var(v4).modifiers().fFlags == SkSL::Modifiers::kFlat_Flag);
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001329
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001330 Var v5(kNoPerspective_Modifier, kInt, "v5");
1331 REPORTER_ASSERT(r, DSLWriter::Var(v5).modifiers().fFlags ==
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001332 SkSL::Modifiers::kNoPerspective_Flag);
1333
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001334 Var v6(kIn_Modifier | kOut_Modifier, kInt, "v6");
1335 REPORTER_ASSERT(r, DSLWriter::Var(v6).modifiers().fFlags ==
1336 (SkSL::Modifiers::kIn_Flag | SkSL::Modifiers::kOut_Flag));
1337
1338 Var v7(kInOut_Modifier, kInt, "v7");
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001339 REPORTER_ASSERT(r, DSLWriter::Var(v7).modifiers().fFlags ==
1340 (SkSL::Modifiers::kIn_Flag | SkSL::Modifiers::kOut_Flag));
1341
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001342 Var v8(kUniform_Modifier, kInt, "v8");
1343 REPORTER_ASSERT(r, DSLWriter::Var(v8).modifiers().fFlags == SkSL::Modifiers::kUniform_Flag);
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001344}
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001345
1346DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLStruct, r, ctxInfo) {
1347 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1348
1349 DSLType simpleStruct = Struct("SimpleStruct",
1350 Field(kFloat, "x"),
1351 Field(kBool, "b"),
1352 Field(Array(kFloat, 3), "a")
1353 );
1354 DSLVar result(simpleStruct, "result");
1355 DSLFunction(simpleStruct, "returnStruct").define(
1356 Declare(result),
1357 result.field("x") = 123,
1358 result.field("b") = result.field("x") > 0,
1359 result.field("a")[0] = result.field("x"),
1360 Return(result)
1361 );
1362 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 2);
John Stilesb4d7b582021-02-19 09:56:31 -05001363 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1364 "struct SimpleStruct { float x; bool b; float[3] a; };");
1365 EXPECT_EQUAL(*DSLWriter::ProgramElements()[1],
1366 "SimpleStruct returnStruct() { SimpleStruct result; (result.x = 123.0);"
1367 "(result.b = (result.x > 0.0)); (result.a[0] = result.x); return result; }");
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001368
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001369 Struct("NestedStruct",
1370 Field(kInt, "x"),
1371 Field(simpleStruct, "simple")
1372 );
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001373 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 3);
1374 EXPECT_EQUAL(*DSLWriter::ProgramElements()[2],
John Stilesb4d7b582021-02-19 09:56:31 -05001375 "struct NestedStruct { int x; SimpleStruct simple; };");
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001376}