blob: a701533e755fd63b5da8fd48b629c17c9c0beebb [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");
338 Expression e1 = a + b;
John Stilesb4d7b582021-02-19 09:56:31 -0500339 EXPECT_EQUAL(e1, "(a + b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500340
341 Expression e2 = a + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500342 EXPECT_EQUAL(e2, "(a + 1.0)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500343
344 Expression e3 = 0.5 + a + -99;
John Stilesb4d7b582021-02-19 09:56:31 -0500345 EXPECT_EQUAL(e3, "((0.5 + a) + -99.0)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500346
347 Expression e4 = a += b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500348 EXPECT_EQUAL(e4, "(a += (b + 1.0))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500349
350 {
351 ExpectError error(r, "error: type mismatch: '+' cannot operate on 'bool2', 'float'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500352 DSLExpression((Bool2(true) + a)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500353 }
354
355 {
356 ExpectError error(r, "error: type mismatch: '+=' cannot operate on 'float', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500357 DSLExpression((a += Bool2(true))).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500358 }
359
360 {
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500361 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500362 DSLExpression((1.0 += a)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500363 }
364}
365
366DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLMinus, r, ctxInfo) {
367 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
368 Var a(kInt, "a"), b(kInt, "b");
369 Expression e1 = a - b;
John Stilesb4d7b582021-02-19 09:56:31 -0500370 EXPECT_EQUAL(e1, "(a - b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500371
372 Expression e2 = a - 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500373 EXPECT_EQUAL(e2, "(a - 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500374
375 Expression e3 = 2 - a - b;
John Stilesb4d7b582021-02-19 09:56:31 -0500376 EXPECT_EQUAL(e3, "((2 - a) - b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500377
378 Expression e4 = a -= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500379 EXPECT_EQUAL(e4, "(a -= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500380
381 {
382 ExpectError error(r, "error: type mismatch: '-' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500383 DSLExpression(Bool2(true) - a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500384 }
385
386 {
387 ExpectError error(r, "error: type mismatch: '-=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500388 DSLExpression(a -= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500389 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500390
391 {
392 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500393 DSLExpression(1.0 -= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500394 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500395}
396
397DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLMultiply, r, ctxInfo) {
398 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
399 Var a(kFloat, "a"), b(kFloat, "b");
400 Expression e1 = a * b;
John Stilesb4d7b582021-02-19 09:56:31 -0500401 EXPECT_EQUAL(e1, "(a * b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500402
403 Expression e2 = a * 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500404 EXPECT_EQUAL(e2, "(a * 1.0)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500405
406 Expression e3 = 0.5 * a * -99;
John Stilesb4d7b582021-02-19 09:56:31 -0500407 EXPECT_EQUAL(e3, "((0.5 * a) * -99.0)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500408
409 Expression e4 = a *= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500410 EXPECT_EQUAL(e4, "(a *= (b + 1.0))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500411
412 {
413 ExpectError error(r, "error: type mismatch: '*' cannot operate on 'bool2', 'float'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500414 DSLExpression(Bool2(true) * a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500415 }
416
417 {
418 ExpectError error(r, "error: type mismatch: '*=' cannot operate on 'float', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500419 DSLExpression(a *= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500420 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500421
422 {
423 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500424 DSLExpression(1.0 *= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500425 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500426}
427
428DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDivide, r, ctxInfo) {
429 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
430 Var a(kFloat, "a"), b(kFloat, "b");
431 Expression e1 = a / b;
John Stilesb4d7b582021-02-19 09:56:31 -0500432 EXPECT_EQUAL(e1, "(a / b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500433
434 Expression e2 = a / 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500435 EXPECT_EQUAL(e2, "(a / 1.0)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500436
437 Expression e3 = 0.5 / a / -99;
John Stilesb4d7b582021-02-19 09:56:31 -0500438 EXPECT_EQUAL(e3, "((0.5 / a) / -99.0)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500439
440 Expression e4 = b / (a - 1);
John Stilesb4d7b582021-02-19 09:56:31 -0500441 EXPECT_EQUAL(e4, "(b / (a - 1.0))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500442
443 Expression e5 = a /= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500444 EXPECT_EQUAL(e5, "(a /= (b + 1.0))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500445
446 {
447 ExpectError error(r, "error: type mismatch: '/' cannot operate on 'bool2', 'float'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500448 DSLExpression(Bool2(true) / a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500449 }
450
451 {
452 ExpectError error(r, "error: type mismatch: '/=' cannot operate on 'float', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500453 DSLExpression(a /= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500454 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500455
456 {
457 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500458 DSLExpression(1.0 /= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500459 }
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500460
461 {
462 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500463 DSLExpression(a /= 0).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500464 }
465
466 {
467 Var c(kFloat2, "c");
468 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500469 DSLExpression(c /= Float2(Float(0), 1)).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500470 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500471}
472
473DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLMod, r, ctxInfo) {
474 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
475 Var a(kInt, "a"), b(kInt, "b");
476 Expression e1 = a % b;
John Stilesb4d7b582021-02-19 09:56:31 -0500477 EXPECT_EQUAL(e1, "(a % b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500478
479 Expression e2 = a % 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500480 EXPECT_EQUAL(e2, "(a % 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500481
482 Expression e3 = 10 % a % -99;
John Stilesb4d7b582021-02-19 09:56:31 -0500483 EXPECT_EQUAL(e3, "((10 % a) % -99)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500484
485 Expression e4 = a %= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500486 EXPECT_EQUAL(e4, "(a %= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500487
488 {
489 ExpectError error(r, "error: type mismatch: '%' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500490 DSLExpression(Bool2(true) % a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500491 }
492
493 {
494 ExpectError error(r, "error: type mismatch: '%=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500495 DSLExpression(a %= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500496 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500497
498 {
499 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500500 DSLExpression(1 %= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500501 }
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500502
503 {
504 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500505 DSLExpression(a %= 0).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500506 }
507
508 {
509 Var c(kInt2, "c");
510 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500511 DSLExpression(c %= Int2(Int(0), 1)).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500512 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500513}
514
515DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLShl, r, ctxInfo) {
516 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
517 Var a(kInt, "a"), b(kInt, "b");
518 Expression e1 = a << b;
John Stilesb4d7b582021-02-19 09:56:31 -0500519 EXPECT_EQUAL(e1, "(a << b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500520
521 Expression e2 = a << 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500522 EXPECT_EQUAL(e2, "(a << 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500523
524 Expression e3 = 1 << a << 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500525 EXPECT_EQUAL(e3, "((1 << a) << 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500526
527 Expression e4 = a <<= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500528 EXPECT_EQUAL(e4, "(a <<= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500529
530 {
531 ExpectError error(r, "error: type mismatch: '<<' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500532 DSLExpression(Bool2(true) << a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500533 }
534
535 {
536 ExpectError error(r, "error: type mismatch: '<<=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500537 DSLExpression(a <<= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500538 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500539
540 {
541 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500542 DSLExpression(1 <<= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500543 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500544}
545
546DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLShr, r, ctxInfo) {
547 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
548 Var a(kInt, "a"), b(kInt, "b");
549 Expression e1 = a >> b;
John Stilesb4d7b582021-02-19 09:56:31 -0500550 EXPECT_EQUAL(e1, "(a >> b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500551
552 Expression e2 = a >> 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500553 EXPECT_EQUAL(e2, "(a >> 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500554
555 Expression e3 = 1 >> a >> 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500556 EXPECT_EQUAL(e3, "((1 >> a) >> 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500557
558 Expression e4 = a >>= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500559 EXPECT_EQUAL(e4, "(a >>= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500560
561 {
562 ExpectError error(r, "error: type mismatch: '>>' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500563 DSLExpression(Bool2(true) >> a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500564 }
565
566 {
567 ExpectError error(r, "error: type mismatch: '>>=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500568 DSLExpression(a >>= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500569 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500570
571 {
572 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500573 DSLExpression(1 >>= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500574 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500575}
576
577DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseAnd, r, ctxInfo) {
578 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
579 Var a(kInt, "a"), b(kInt, "b");
580 Expression e1 = a & b;
John Stilesb4d7b582021-02-19 09:56:31 -0500581 EXPECT_EQUAL(e1, "(a & b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500582
583 Expression e2 = a & 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500584 EXPECT_EQUAL(e2, "(a & 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500585
586 Expression e3 = 1 & a & 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500587 EXPECT_EQUAL(e3, "((1 & a) & 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500588
589 Expression e4 = a &= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500590 EXPECT_EQUAL(e4, "(a &= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500591
592 {
593 ExpectError error(r, "error: type mismatch: '&' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500594 DSLExpression(Bool2(true) & a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500595 }
596
597 {
598 ExpectError error(r, "error: type mismatch: '&=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500599 DSLExpression(a &= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500600 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500601
602 {
603 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500604 DSLExpression(1 &= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500605 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500606}
607
608DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseOr, r, ctxInfo) {
609 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
610 Var a(kInt, "a"), b(kInt, "b");
611 Expression e1 = a | b;
John Stilesb4d7b582021-02-19 09:56:31 -0500612 EXPECT_EQUAL(e1, "(a | b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500613
614 Expression e2 = a | 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500615 EXPECT_EQUAL(e2, "(a | 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500616
617 Expression e3 = 1 | a | 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500618 EXPECT_EQUAL(e3, "((1 | a) | 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500619
620 Expression e4 = a |= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500621 EXPECT_EQUAL(e4, "(a |= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500622
623 {
624 ExpectError error(r, "error: type mismatch: '|' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500625 DSLExpression(Bool2(true) | a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500626 }
627
628 {
629 ExpectError error(r, "error: type mismatch: '|=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500630 DSLExpression(a |= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500631 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500632
633 {
634 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500635 DSLExpression(1 |= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500636 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500637}
638
639DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseXor, r, ctxInfo) {
640 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
641 Var a(kInt, "a"), b(kInt, "b");
642 Expression e1 = a ^ b;
John Stilesb4d7b582021-02-19 09:56:31 -0500643 EXPECT_EQUAL(e1, "(a ^ b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500644
645 Expression e2 = a ^ 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500646 EXPECT_EQUAL(e2, "(a ^ 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500647
648 Expression e3 = 1 ^ a ^ 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500649 EXPECT_EQUAL(e3, "((1 ^ a) ^ 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500650
651 Expression e4 = a ^= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500652 EXPECT_EQUAL(e4, "(a ^= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500653
654 {
655 ExpectError error(r, "error: type mismatch: '^' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500656 DSLExpression(Bool2(true) ^ a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500657 }
658
659 {
660 ExpectError error(r, "error: type mismatch: '^=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500661 DSLExpression(a ^= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500662 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500663
664 {
665 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500666 DSLExpression(1 ^= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500667 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500668}
669
670DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLogicalAnd, r, ctxInfo) {
671 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
672 Var a(kBool, "a"), b(kBool, "b");
673 Expression e1 = a && b;
John Stilesb4d7b582021-02-19 09:56:31 -0500674 EXPECT_EQUAL(e1, "(a && b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500675
676 Expression e2 = a && true && b;
John Stilesb4d7b582021-02-19 09:56:31 -0500677 EXPECT_EQUAL(e2, "(a && b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500678
679 Expression e3 = a && false && b;
John Stilesb4d7b582021-02-19 09:56:31 -0500680 EXPECT_EQUAL(e3, "false");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500681
682 {
683 ExpectError error(r, "error: type mismatch: '&&' cannot operate on 'bool', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500684 DSLExpression(a && 5).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500685 }
686}
687
688DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLogicalOr, r, ctxInfo) {
689 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
690 Var a(kBool, "a"), b(kBool, "b");
691 Expression e1 = a || b;
John Stilesb4d7b582021-02-19 09:56:31 -0500692 EXPECT_EQUAL(e1, "(a || b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500693
694 Expression e2 = a || true || b;
John Stilesb4d7b582021-02-19 09:56:31 -0500695 EXPECT_EQUAL(e2, "true");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500696
697 Expression e3 = a || false || b;
John Stilesb4d7b582021-02-19 09:56:31 -0500698 EXPECT_EQUAL(e3, "(a || b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500699
700 {
701 ExpectError error(r, "error: type mismatch: '||' cannot operate on 'bool', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500702 DSLExpression(a || 5).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500703 }
704}
705
706DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLComma, r, ctxInfo) {
707 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
708 Var a(kInt, "a"), b(kInt, "b");
709 Expression e1 = (a += b, b);
John Stilesb4d7b582021-02-19 09:56:31 -0500710 EXPECT_EQUAL(e1, "((a += b) , b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500711
712 Expression e2 = (a += b, b += b, Int2(a));
John Stilesb4d7b582021-02-19 09:56:31 -0500713 EXPECT_EQUAL(e2, "(((a += b) , (b += b)) , int2(a))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500714}
715
716DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLEqual, r, ctxInfo) {
717 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
718 Var a(kInt, "a"), b(kInt, "b");
719 Expression e1 = a == b;
John Stilesb4d7b582021-02-19 09:56:31 -0500720 EXPECT_EQUAL(e1, "(a == b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500721
722 Expression e2 = a == 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500723 EXPECT_EQUAL(e2, "(a == 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500724
725 {
726 ExpectError error(r, "error: type mismatch: '==' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500727 DSLExpression(a == Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500728 }
729}
730
731DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLNotEqual, r, ctxInfo) {
732 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
733 Var a(kInt, "a"), b(kInt, "b");
734 Expression e1 = a != b;
John Stilesb4d7b582021-02-19 09:56:31 -0500735 EXPECT_EQUAL(e1, "(a != b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500736
737 Expression e2 = a != 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500738 EXPECT_EQUAL(e2, "(a != 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500739
740 {
741 ExpectError error(r, "error: type mismatch: '!=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500742 DSLExpression(a != Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500743 }
744}
745
746DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLGreaterThan, r, ctxInfo) {
747 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
748 Var a(kInt, "a"), b(kInt, "b");
749 Expression e1 = a > b;
John Stilesb4d7b582021-02-19 09:56:31 -0500750 EXPECT_EQUAL(e1, "(a > b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500751
752 Expression e2 = a > 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500753 EXPECT_EQUAL(e2, "(a > 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500754
755 {
756 ExpectError error(r, "error: type mismatch: '>' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500757 DSLExpression(a > Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500758 }
759}
760
761DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLGreaterThanOrEqual, r, ctxInfo) {
762 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
763 Var a(kInt, "a"), b(kInt, "b");
764 Expression e1 = a >= b;
John Stilesb4d7b582021-02-19 09:56:31 -0500765 EXPECT_EQUAL(e1, "(a >= b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500766
767 Expression e2 = a >= 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500768 EXPECT_EQUAL(e2, "(a >= 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500769
770 {
771 ExpectError error(r, "error: type mismatch: '>=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500772 DSLExpression(a >= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500773 }
774}
775
776DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLessThan, r, ctxInfo) {
777 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
778 Var a(kInt, "a"), b(kInt, "b");
779 Expression e1 = a < b;
John Stilesb4d7b582021-02-19 09:56:31 -0500780 EXPECT_EQUAL(e1, "(a < b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500781
782 Expression e2 = a < 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500783 EXPECT_EQUAL(e2, "(a < 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500784
785 {
786 ExpectError error(r, "error: type mismatch: '<' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500787 DSLExpression(a < Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500788 }
789}
790
791DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLessThanOrEqual, r, ctxInfo) {
792 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
793 Var a(kInt, "a"), b(kInt, "b");
794 Expression e1 = a <= b;
John Stilesb4d7b582021-02-19 09:56:31 -0500795 EXPECT_EQUAL(e1, "(a <= b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500796
797 Expression e2 = a <= 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500798 EXPECT_EQUAL(e2, "(a <= 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500799
800 {
801 ExpectError error(r, "error: type mismatch: '<=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500802 DSLExpression(a <= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500803 }
804}
805
806DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLogicalNot, r, ctxInfo) {
807 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
808 Var a(kInt, "a"), b(kInt, "b");
809 Expression e1 = !(a <= b);
John Stilesb4d7b582021-02-19 09:56:31 -0500810 EXPECT_EQUAL(e1, "!(a <= b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500811
812 {
813 ExpectError error(r, "error: '!' cannot operate on 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500814 DSLExpression(!a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500815 }
816}
817
818DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseNot, r, ctxInfo) {
819 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
820 Var a(kInt, "a"), b(kBool, "b");
821 Expression e1 = ~a;
John Stilesb4d7b582021-02-19 09:56:31 -0500822 EXPECT_EQUAL(e1, "~a");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500823
824 {
825 ExpectError error(r, "error: '~' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500826 DSLExpression(~b).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500827 }
828}
829
830DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLIncrement, r, ctxInfo) {
831 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
832 Var a(kInt, "a"), b(kBool, "b");
833 Expression e1 = ++a;
John Stilesb4d7b582021-02-19 09:56:31 -0500834 EXPECT_EQUAL(e1, "++a");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500835
836 Expression e2 = a++;
John Stilesb4d7b582021-02-19 09:56:31 -0500837 EXPECT_EQUAL(e2, "a++");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500838
839 {
840 ExpectError error(r, "error: '++' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500841 DSLExpression(++b).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500842 }
843
844 {
845 ExpectError error(r, "error: '++' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500846 DSLExpression(b++).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500847 }
848
849 {
850 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500851 DSLExpression(++(a + 1)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500852 }
853
854 {
855 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500856 DSLExpression((a + 1)++).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500857 }
858}
859
860DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDecrement, r, ctxInfo) {
861 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
862 Var a(kInt, "a"), b(kBool, "b");
863 Expression e1 = --a;
John Stilesb4d7b582021-02-19 09:56:31 -0500864 EXPECT_EQUAL(e1, "--a");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500865
866 Expression e2 = a--;
John Stilesb4d7b582021-02-19 09:56:31 -0500867 EXPECT_EQUAL(e2, "a--");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500868
869 {
870 ExpectError error(r, "error: '--' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500871 DSLExpression(--b).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500872 }
873
874 {
875 ExpectError error(r, "error: '--' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500876 DSLExpression(b--).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500877 }
878
879 {
880 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500881 DSLExpression(--(a + 1)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500882 }
883
884 {
885 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500886 DSLExpression((a + 1)--).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500887 }
888}
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500889
890DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBlock, r, ctxInfo) {
891 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
892 Statement x = Block();
John Stilesb4d7b582021-02-19 09:56:31 -0500893 EXPECT_EQUAL(x, "{ }");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500894 Var a(kInt, "a", 1), b(kInt, "b", 2);
895 Statement y = Block(Declare(a), Declare(b), a = b);
John Stilesb4d7b582021-02-19 09:56:31 -0500896 EXPECT_EQUAL(y, "{ int a = 1; int b = 2; (a = b); }");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500897}
898
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500899DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBreak, r, ctxInfo) {
900 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500901 Var i(kInt, "i", 0);
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500902 DSLFunction(kVoid, "success").define(
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500903 For(Declare(i), i < 10, ++i, Block(
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500904 If(i > 5, Break())
905 ))
906 );
907 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
John Stilesb4d7b582021-02-19 09:56:31 -0500908 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
909 "void success() { for (int i = 0; (i < 10); ++i) { if ((i > 5)) break; } }");
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500910
911 {
912 ExpectError error(r, "error: break statement must be inside a loop or switch\n");
913 DSLFunction(kVoid, "fail").define(
914 Break()
915 );
916 }
917}
918
919DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLContinue, r, ctxInfo) {
920 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500921 Var i(kInt, "i", 0);
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500922 DSLFunction(kVoid, "success").define(
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500923 For(Declare(i), i < 10, ++i, Block(
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500924 If(i < 5, Continue())
925 ))
926 );
927 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
John Stilesb4d7b582021-02-19 09:56:31 -0500928 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
929 "void success() { for (int i = 0; (i < 10); ++i) { if ((i < 5)) continue; } }");
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500930
931 {
932 ExpectError error(r, "error: continue statement must be inside a loop\n");
933 DSLFunction(kVoid, "fail").define(
934 Continue()
935 );
936 }
937}
938
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500939DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDeclare, r, ctxInfo) {
940 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500941 Var a(kHalf4, "a"), b(kHalf4, "b", Half4(1));
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500942 Statement x = Declare(a);
John Stilesb4d7b582021-02-19 09:56:31 -0500943 EXPECT_EQUAL(x, "half4 a;");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500944 Statement y = Declare(b);
John Stilesb4d7b582021-02-19 09:56:31 -0500945 EXPECT_EQUAL(y, "half4 b = half4(1.0);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500946
947 {
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500948 Var c(kHalf4, "c", 1);
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500949 ExpectError error(r, "error: expected 'half4', but found 'int'\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500950 Declare(c).release();
951 }
952
953 {
954 Var d(kInt, "d");
955 Declare(d).release();
956 ExpectError error(r, "error: variable has already been declared\n");
957 Declare(d).release();
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500958 }
959}
960
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500961DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDiscard, r, ctxInfo) {
962 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
963 Statement x = If(Sqrt(1) > 0, Discard());
John Stilesb4d7b582021-02-19 09:56:31 -0500964 EXPECT_EQUAL(x, "if ((sqrt(1.0) > 0.0)) discard;");
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500965}
966
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500967DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDo, r, ctxInfo) {
968 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
969 Statement x = Do(Block(), true);
John Stilesb4d7b582021-02-19 09:56:31 -0500970 EXPECT_EQUAL(x, "do {} while (true);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500971
972 Var a(kFloat, "a"), b(kFloat, "b");
973 Statement y = Do(Block(a++, --b), a != b);
John Stilesb4d7b582021-02-19 09:56:31 -0500974 EXPECT_EQUAL(y, "do { a++; --b; } while ((a != b));");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500975
976 {
977 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
978 Do(Block(), 7).release();
979 }
980}
981
982DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLFor, r, ctxInfo) {
983 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
984 Statement x = For(Statement(), Expression(), Expression(), Block());
John Stilesb4d7b582021-02-19 09:56:31 -0500985 EXPECT_EQUAL(x, "for (;;) {}");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500986
Ethan Nicholasfe5d6922021-03-05 14:23:48 -0500987 Var i(kInt, "i", 0);
988 Statement y = For(Declare(i), i < 10, ++i, i += 5);
John Stilesb4d7b582021-02-19 09:56:31 -0500989 EXPECT_EQUAL(y, "for (int i = 0; (i < 10); ++i) (i += 5);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500990
991 {
992 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
993 For(i = 0, i + 10, ++i, i += 5).release();
994 }
995}
996
Ethan Nicholase2c05042021-02-03 10:27:22 -0500997DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLFunction, r, ctxInfo) {
Ethan Nicholas1ff76092021-01-28 10:02:43 -0500998 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholas1ff76092021-01-28 10:02:43 -0500999 Var coords(kHalf2, "coords");
1000 DSLFunction(kVoid, "main", coords).define(
1001 sk_FragColor() = Half4(coords, 0, 1)
1002 );
1003 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
John Stilesb4d7b582021-02-19 09:56:31 -05001004 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1005 "void main(half2 coords) { (sk_FragColor = half4(coords, 0.0, 1.0)); }");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001006
Ethan Nicholas63f75fc2021-02-23 12:05:49 -05001007 {
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001008 DSLWriter::Reset();
Ethan Nicholas63f75fc2021-02-23 12:05:49 -05001009 Var x(kFloat, "x");
1010 DSLFunction sqr(kFloat, "sqr", x);
1011 sqr.define(
1012 Return(x * x)
1013 );
1014 EXPECT_EQUAL(sqr(sk_FragCoord().x()), "sqr(sk_FragCoord.x)");
1015 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
1016 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0], "float sqr(float x) { return (x * x); }");
1017 }
1018
1019 {
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001020 DSLWriter::Reset();
Ethan Nicholas63f75fc2021-02-23 12:05:49 -05001021 Var x(kFloat2, "x");
1022 Var y(kFloat2, "y");
1023 DSLFunction dot(kFloat2, "dot", x, y);
1024 dot.define(
1025 Return(x * x + y * y)
1026 );
1027 EXPECT_EQUAL(dot(Float2(1.0f, 2.0f), Float2(3.0f, 4.0f)),
1028 "dot(float2(1.0, 2.0), float2(3.0, 4.0))");
1029 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
1030 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1031 "float2 dot(float2 x, float2 y) { return ((x * x) + (y * y)); }");
1032 }
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001033
1034 {
1035 ExpectError error(r, "error: expected 'float', but found 'bool'\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001036 DSLWriter::Reset();
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001037 DSLFunction(kFloat, "broken").define(
1038 Return(true)
1039 );
1040 }
1041
1042 {
John Stilesb3dcbb12021-03-04 16:00:20 -05001043 ExpectError error(r, "error: expected function to return 'float'\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001044 DSLWriter::Reset();
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001045 DSLFunction(kFloat, "broken").define(
1046 Return()
1047 );
1048 }
1049
1050 {
John Stilesb3dcbb12021-03-04 16:00:20 -05001051 ExpectError error(r, "error: function 'broken' can exit without returning a value\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001052 DSLWriter::Reset();
1053 Var x(kFloat, "x", 0);
John Stilesb3dcbb12021-03-04 16:00:20 -05001054 DSLFunction(kFloat, "broken").define(
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001055 Declare(x),
John Stilesb3dcbb12021-03-04 16:00:20 -05001056 If(x == 1, Return(x))
1057 );
1058 }
1059
1060 {
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001061 ExpectError error(r, "error: may not return a value from a void function\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001062 DSLWriter::Reset();
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001063 DSLFunction(kVoid, "broken").define(
1064 Return(0)
1065 );
1066 }
1067
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001068 {
John Stilesb3dcbb12021-03-04 16:00:20 -05001069 ExpectError error(r, "error: function 'broken' can exit without returning a value\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001070 DSLWriter::Reset();
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001071 DSLFunction(kFloat, "broken").define(
1072 );
1073 }
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001074}
1075
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001076DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLIf, r, ctxInfo) {
1077 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1078 Var a(kFloat, "a"), b(kFloat, "b");
1079 Statement x = If(a > b, a -= b);
John Stilesb4d7b582021-02-19 09:56:31 -05001080 EXPECT_EQUAL(x, "if ((a > b)) (a -= b);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001081
1082 Statement y = If(a > b, a -= b, b -= a);
John Stilesb4d7b582021-02-19 09:56:31 -05001083 EXPECT_EQUAL(y, "if ((a > b)) (a -= b); else (b -= a);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001084
1085 {
1086 ExpectError error(r, "error: expected 'bool', but found 'float'\n");
1087 If(a + b, a -= b).release();
1088 }
1089}
1090
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001091DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLReturn, r, ctxInfo) {
1092 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1093
1094 Statement x = Return();
John Stilesb4d7b582021-02-19 09:56:31 -05001095 EXPECT_EQUAL(x, "return;");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001096
1097 Statement y = Return(true);
John Stilesb4d7b582021-02-19 09:56:31 -05001098 EXPECT_EQUAL(y, "return true;");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001099}
1100
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001101DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSelect, r, ctxInfo) {
1102 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1103 Var a(kInt, "a");
1104 Expression x = Select(a > 0, 1, -1);
John Stilesb4d7b582021-02-19 09:56:31 -05001105 EXPECT_EQUAL(x, "((a > 0) ? 1 : -1)");
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001106
1107 {
1108 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001109 DSLExpression x = Select(a, 1, -1);
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001110 }
1111
1112 {
1113 ExpectError error(r, "error: ternary operator result mismatch: 'float2', 'float3'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001114 DSLExpression x = Select(a > 0, Float2(1), Float3(1));
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001115 }
1116}
1117
Ethan Nicholascfefec02021-02-09 15:22:57 -05001118DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSwitch, r, ctxInfo) {
1119 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1120
1121 Var a(kFloat, "a"), b(kInt, "b");
1122
1123 Statement x = Switch(5,
1124 Case(0, a = 0, Break()),
1125 Case(1, a = 1, Continue()),
John Stilese1d1b082021-02-23 13:44:36 -05001126 Case(2, a = 2 /*Fallthrough*/),
Ethan Nicholascfefec02021-02-09 15:22:57 -05001127 Default(Discard())
1128 );
John Stilese1d1b082021-02-23 13:44:36 -05001129 EXPECT_EQUAL(x, R"(
1130 switch (5) {
1131 case 0: (a = 0.0); break;
1132 case 1: (a = 1.0); continue;
1133 case 2: (a = 2.0);
1134 default: discard;
1135 }
1136 )");
Ethan Nicholascfefec02021-02-09 15:22:57 -05001137
John Stiles642cde22021-02-23 14:57:01 -05001138 EXPECT_EQUAL(Switch(b),
1139 "switch (b) {}");
1140
1141 EXPECT_EQUAL(Switch(b, Default(), Case(0), Case(1)),
1142 "switch (b) { default: case 0: case 1: }");
Ethan Nicholascfefec02021-02-09 15:22:57 -05001143
1144 {
John Stilese1d1b082021-02-23 13:44:36 -05001145 ExpectError error(r, "error: duplicate case value '0'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001146 DSLStatement(Switch(0, Case(0), Case(0))).release();
Ethan Nicholascfefec02021-02-09 15:22:57 -05001147 }
1148
1149 {
John Stilese1d1b082021-02-23 13:44:36 -05001150 ExpectError error(r, "error: duplicate default case\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001151 DSLStatement(Switch(0, Default(a = 0), Default(a = 1))).release();
John Stilese1d1b082021-02-23 13:44:36 -05001152 }
1153
1154 {
Ethan Nicholascfefec02021-02-09 15:22:57 -05001155 ExpectError error(r, "error: case value must be a constant integer\n");
1156 Var b(kInt);
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001157 DSLStatement(Switch(0, Case(b))).release();
Ethan Nicholascfefec02021-02-09 15:22:57 -05001158 }
1159
1160 {
1161 ExpectError error(r, "error: continue statement must be inside a loop\n");
1162 DSLFunction(kVoid, "fail").define(
1163 Switch(5,
1164 Case(0, a = 0, Break()),
1165 Case(1, a = 1, Continue()),
1166 Default(Discard())
1167 )
1168 );
1169 }
1170}
1171
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001172DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSwizzle, r, ctxInfo) {
1173 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1174 Var a(kFloat4, "a");
1175
John Stilesf04e09c2021-03-05 13:13:14 -05001176 EXPECT_EQUAL(a.x(),
1177 "a.x");
1178 EXPECT_EQUAL(a.y(),
1179 "a.y");
1180 EXPECT_EQUAL(a.z(),
1181 "a.z");
1182 EXPECT_EQUAL(a.w(),
1183 "a.w");
1184 EXPECT_EQUAL(a.r(),
1185 "a.x");
1186 EXPECT_EQUAL(a.g(),
1187 "a.y");
1188 EXPECT_EQUAL(a.b(),
1189 "a.z");
1190 EXPECT_EQUAL(a.a(),
1191 "a.w");
1192 EXPECT_EQUAL(Swizzle(a, R),
1193 "a.x");
1194 EXPECT_EQUAL(Swizzle(a, ZERO, G),
1195 "float2(0.0, a.y)");
1196 EXPECT_EQUAL(Swizzle(a, B, G, G),
1197 "a.zyy");
1198 EXPECT_EQUAL(Swizzle(a, R, G, B, ONE),
1199 "float4(a.xyz, 1.0)");
1200 EXPECT_EQUAL(Swizzle(a, B, G, R, ONE).r(),
1201 "a.z");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001202}
1203
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001204DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLWhile, r, ctxInfo) {
1205 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1206 Statement x = While(true, Block());
John Stilesb4d7b582021-02-19 09:56:31 -05001207 EXPECT_EQUAL(x, "for (; true;) {}");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001208
1209 Var a(kFloat, "a"), b(kFloat, "b");
1210 Statement y = While(a != b, Block(a++, --b));
John Stilesb4d7b582021-02-19 09:56:31 -05001211 EXPECT_EQUAL(y, "for (; (a != b);) { a++; --b; }");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001212
1213 {
1214 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001215 DSLStatement x = While(7, Block());
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001216 }
1217}
Ethan Nicholas04be3392021-01-26 10:07:01 -05001218
1219DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLIndex, r, ctxInfo) {
1220 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1221 Var a(Array(kInt, 5), "a"), b(kInt, "b");
John Stilesb4d7b582021-02-19 09:56:31 -05001222
1223 EXPECT_EQUAL(a[0], "a[0]");
1224 EXPECT_EQUAL(a[b], "a[b]");
Ethan Nicholas04be3392021-01-26 10:07:01 -05001225
1226 {
1227 ExpectError error(r, "error: expected 'int', but found 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001228 DSLExpression x = a[true];
Ethan Nicholas04be3392021-01-26 10:07:01 -05001229 }
1230
1231 {
1232 ExpectError error(r, "error: expected array, but found 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001233 DSLExpression x = b[0];
Ethan Nicholas04be3392021-01-26 10:07:01 -05001234 }
1235
1236 {
1237 ExpectError error(r, "error: index -1 out of range for 'int[5]'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001238 DSLExpression x = a[-1];
Ethan Nicholas04be3392021-01-26 10:07:01 -05001239 }
1240}
Ethan Nicholas30e93d52021-01-26 12:00:25 -05001241
1242DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBuiltins, r, ctxInfo) {
1243 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1244 // There is a Fract type on Mac which can conflict with our Fract builtin
1245 using SkSL::dsl::Fract;
1246 Var a(kHalf4, "a"), b(kHalf4, "b"), c(kHalf4, "c");
1247 Var h3(kHalf3, "h3");
1248 Var b4(kBool4, "b4");
John Stilesb4d7b582021-02-19 09:56:31 -05001249 EXPECT_EQUAL(Abs(a), "abs(a)");
1250 EXPECT_EQUAL(All(b4), "all(b4)");
1251 EXPECT_EQUAL(Any(b4), "any(b4)");
1252 EXPECT_EQUAL(Ceil(a), "ceil(a)");
1253 EXPECT_EQUAL(Clamp(a, 0, 1), "clamp(a, 0.0, 1.0)");
1254 EXPECT_EQUAL(Cos(a), "cos(a)");
1255 EXPECT_EQUAL(Cross(h3, h3), "cross(h3, h3)");
1256 EXPECT_EQUAL(Degrees(a), "degrees(a)");
1257 EXPECT_EQUAL(Distance(a, b), "distance(a, b)");
1258 EXPECT_EQUAL(Dot(a, b), "dot(a, b)");
1259 EXPECT_EQUAL(Equal(a, b), "equal(a, b)");
1260 EXPECT_EQUAL(Exp(a), "exp(a)");
1261 EXPECT_EQUAL(Exp2(a), "exp2(a)");
1262 EXPECT_EQUAL(Faceforward(a, b, c), "faceforward(a, b, c)");
1263 EXPECT_EQUAL(Floor(a), "floor(a)");
1264 EXPECT_EQUAL(Fract(a), "fract(a)");
1265 EXPECT_EQUAL(GreaterThan(a, b), "greaterThan(a, b)");
1266 EXPECT_EQUAL(GreaterThanEqual(a, b), "greaterThanEqual(a, b)");
1267 EXPECT_EQUAL(Inversesqrt(a), "inversesqrt(a)");
1268 EXPECT_EQUAL(LessThan(a, b), "lessThan(a, b)");
1269 EXPECT_EQUAL(LessThanEqual(a, b), "lessThanEqual(a, b)");
1270 EXPECT_EQUAL(Length(a), "length(a)");
1271 EXPECT_EQUAL(Log(a), "log(a)");
1272 EXPECT_EQUAL(Log2(a), "log2(a)");
1273 EXPECT_EQUAL(Max(a, b), "max(a, b)");
1274 EXPECT_EQUAL(Min(a, b), "min(a, b)");
1275 EXPECT_EQUAL(Mix(a, b, c), "mix(a, b, c)");
1276 EXPECT_EQUAL(Mod(a, b), "mod(a, b)");
1277 EXPECT_EQUAL(Normalize(a), "normalize(a)");
1278 EXPECT_EQUAL(NotEqual(a, b), "notEqual(a, b)");
1279 EXPECT_EQUAL(Pow(a, b), "pow(a, b)");
1280 EXPECT_EQUAL(Radians(a), "radians(a)");
1281 EXPECT_EQUAL(Reflect(a, b), "reflect(a, b)");
1282 EXPECT_EQUAL(Refract(a, b, 1), "refract(a, b, 1.0)");
1283 EXPECT_EQUAL(Saturate(a), "saturate(a)");
1284 EXPECT_EQUAL(Sign(a), "sign(a)");
1285 EXPECT_EQUAL(Sin(a), "sin(a)");
1286 EXPECT_EQUAL(Smoothstep(a, b, c), "smoothstep(a, b, c)");
1287 EXPECT_EQUAL(Sqrt(a), "sqrt(a)");
1288 EXPECT_EQUAL(Step(a, b), "step(a, b)");
1289 EXPECT_EQUAL(Tan(a), "tan(a)");
1290 EXPECT_EQUAL(Unpremul(a), "unpremul(a)");
Ethan Nicholas30e93d52021-01-26 12:00:25 -05001291
1292 // these calls all go through the normal channels, so it ought to be sufficient to prove that
1293 // one of them reports errors correctly
1294 {
1295 ExpectError error(r, "error: no match for ceil(bool)\n");
1296 Ceil(a == b).release();
1297 }
1298}
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001299
1300DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLModifiers, r, ctxInfo) {
1301 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1302
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001303 Var v1(kConst_Modifier, kInt, "v1", 0);
1304 Statement d1 = Declare(v1);
Ethan Nicholasbd974002021-02-22 16:20:06 -05001305 EXPECT_EQUAL(d1, "const int v1 = 0;");
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001306
1307 // Most modifiers require an appropriate context to be legal. We can't yet give them that
1308 // context, so we can't as yet Declare() variables with these modifiers.
1309 // TODO: better tests when able
1310 Var v2(kIn_Modifier, kInt, "v2");
1311 REPORTER_ASSERT(r, DSLWriter::Var(v2).modifiers().fFlags == SkSL::Modifiers::kIn_Flag);
1312
1313 Var v3(kOut_Modifier, kInt, "v3");
1314 REPORTER_ASSERT(r, DSLWriter::Var(v3).modifiers().fFlags == SkSL::Modifiers::kOut_Flag);
1315
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001316 Var v4(kFlat_Modifier, kInt, "v4");
1317 REPORTER_ASSERT(r, DSLWriter::Var(v4).modifiers().fFlags == SkSL::Modifiers::kFlat_Flag);
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001318
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001319 Var v5(kNoPerspective_Modifier, kInt, "v5");
1320 REPORTER_ASSERT(r, DSLWriter::Var(v5).modifiers().fFlags ==
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001321 SkSL::Modifiers::kNoPerspective_Flag);
1322
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001323 Var v6(kIn_Modifier | kOut_Modifier, kInt, "v6");
1324 REPORTER_ASSERT(r, DSLWriter::Var(v6).modifiers().fFlags ==
1325 (SkSL::Modifiers::kIn_Flag | SkSL::Modifiers::kOut_Flag));
1326
1327 Var v7(kInOut_Modifier, kInt, "v7");
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001328 REPORTER_ASSERT(r, DSLWriter::Var(v7).modifiers().fFlags ==
1329 (SkSL::Modifiers::kIn_Flag | SkSL::Modifiers::kOut_Flag));
1330
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001331 Var v8(kUniform_Modifier, kInt, "v8");
1332 REPORTER_ASSERT(r, DSLWriter::Var(v8).modifiers().fFlags == SkSL::Modifiers::kUniform_Flag);
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001333}
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001334
1335DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLStruct, r, ctxInfo) {
1336 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1337
1338 DSLType simpleStruct = Struct("SimpleStruct",
1339 Field(kFloat, "x"),
1340 Field(kBool, "b"),
1341 Field(Array(kFloat, 3), "a")
1342 );
1343 DSLVar result(simpleStruct, "result");
1344 DSLFunction(simpleStruct, "returnStruct").define(
1345 Declare(result),
1346 result.field("x") = 123,
1347 result.field("b") = result.field("x") > 0,
1348 result.field("a")[0] = result.field("x"),
1349 Return(result)
1350 );
1351 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 2);
John Stilesb4d7b582021-02-19 09:56:31 -05001352 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1353 "struct SimpleStruct { float x; bool b; float[3] a; };");
1354 EXPECT_EQUAL(*DSLWriter::ProgramElements()[1],
1355 "SimpleStruct returnStruct() { SimpleStruct result; (result.x = 123.0);"
1356 "(result.b = (result.x > 0.0)); (result.a[0] = result.x); return result; }");
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001357
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001358 Struct("NestedStruct",
1359 Field(kInt, "x"),
1360 Field(simpleStruct, "simple")
1361 );
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001362 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 3);
1363 EXPECT_EQUAL(*DSLWriter::ProgramElements()[2],
John Stilesb4d7b582021-02-19 09:56:31 -05001364 "struct NestedStruct { int x; SimpleStruct simple; };");
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001365}