blob: b5b8b7f5b6beb703d51c78ec2585f961be52915d [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 Nicholasd6b6f3e2021-01-22 15:18:25 -0500894 Var a(kInt, "a"), b(kInt, "b");
895 Statement y = Block(Declare(a, 1), Declare(b, 2), 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());
901 Var i(kInt, "i");
902 DSLFunction(kVoid, "success").define(
903 For(Declare(i, 0), i < 10, ++i, Block(
904 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());
921 Var i(kInt, "i");
922 DSLFunction(kVoid, "success").define(
923 For(Declare(i, 0), i < 10, ++i, Block(
924 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());
941 Var a(kHalf4, "a"), b(kHalf4, "b");
942 Statement x = Declare(a);
John Stilesb4d7b582021-02-19 09:56:31 -0500943 EXPECT_EQUAL(x, "half4 a;");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500944 Statement y = Declare(b, Half4(1));
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 {
948 Var c(kHalf4, "c");
949 ExpectError error(r, "error: expected 'half4', but found 'int'\n");
950 Declare(c, 1).release();
951 }
952}
953
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500954DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDiscard, r, ctxInfo) {
955 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
956 Statement x = If(Sqrt(1) > 0, Discard());
John Stilesb4d7b582021-02-19 09:56:31 -0500957 EXPECT_EQUAL(x, "if ((sqrt(1.0) > 0.0)) discard;");
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500958}
959
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500960DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDo, r, ctxInfo) {
961 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
962 Statement x = Do(Block(), true);
John Stilesb4d7b582021-02-19 09:56:31 -0500963 EXPECT_EQUAL(x, "do {} while (true);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500964
965 Var a(kFloat, "a"), b(kFloat, "b");
966 Statement y = Do(Block(a++, --b), a != b);
John Stilesb4d7b582021-02-19 09:56:31 -0500967 EXPECT_EQUAL(y, "do { a++; --b; } while ((a != b));");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500968
969 {
970 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
971 Do(Block(), 7).release();
972 }
973}
974
975DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLFor, r, ctxInfo) {
976 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
977 Statement x = For(Statement(), Expression(), Expression(), Block());
John Stilesb4d7b582021-02-19 09:56:31 -0500978 EXPECT_EQUAL(x, "for (;;) {}");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500979
980 Var i(kInt, "i");
981 Statement y = For(Declare(i, 0), i < 10, ++i, i += 5);
John Stilesb4d7b582021-02-19 09:56:31 -0500982 EXPECT_EQUAL(y, "for (int i = 0; (i < 10); ++i) (i += 5);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500983
984 {
985 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
986 For(i = 0, i + 10, ++i, i += 5).release();
987 }
988}
989
Ethan Nicholase2c05042021-02-03 10:27:22 -0500990DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLFunction, r, ctxInfo) {
Ethan Nicholas1ff76092021-01-28 10:02:43 -0500991 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
992 DSLWriter::ProgramElements().clear();
993 Var coords(kHalf2, "coords");
994 DSLFunction(kVoid, "main", coords).define(
995 sk_FragColor() = Half4(coords, 0, 1)
996 );
997 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
John Stilesb4d7b582021-02-19 09:56:31 -0500998 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
999 "void main(half2 coords) { (sk_FragColor = half4(coords, 0.0, 1.0)); }");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001000
Ethan Nicholas63f75fc2021-02-23 12:05:49 -05001001 {
1002 DSLWriter::ProgramElements().clear();
1003 Var x(kFloat, "x");
1004 DSLFunction sqr(kFloat, "sqr", x);
1005 sqr.define(
1006 Return(x * x)
1007 );
1008 EXPECT_EQUAL(sqr(sk_FragCoord().x()), "sqr(sk_FragCoord.x)");
1009 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
1010 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0], "float sqr(float x) { return (x * x); }");
1011 }
1012
1013 {
1014 DSLWriter::ProgramElements().clear();
1015 Var x(kFloat2, "x");
1016 Var y(kFloat2, "y");
1017 DSLFunction dot(kFloat2, "dot", x, y);
1018 dot.define(
1019 Return(x * x + y * y)
1020 );
1021 EXPECT_EQUAL(dot(Float2(1.0f, 2.0f), Float2(3.0f, 4.0f)),
1022 "dot(float2(1.0, 2.0), float2(3.0, 4.0))");
1023 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
1024 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1025 "float2 dot(float2 x, float2 y) { return ((x * x) + (y * y)); }");
1026 }
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001027
1028 {
1029 ExpectError error(r, "error: expected 'float', but found 'bool'\n");
1030 DSLWriter::ProgramElements().clear();
1031 DSLFunction(kFloat, "broken").define(
1032 Return(true)
1033 );
1034 }
1035
1036 {
John Stilesb3dcbb12021-03-04 16:00:20 -05001037 ExpectError error(r, "error: expected function to return 'float'\n");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001038 DSLWriter::ProgramElements().clear();
1039 DSLFunction(kFloat, "broken").define(
1040 Return()
1041 );
1042 }
1043
1044 {
John Stilesb3dcbb12021-03-04 16:00:20 -05001045 ExpectError error(r, "error: function 'broken' can exit without returning a value\n");
1046 DSLWriter::ProgramElements().clear();
1047 Var x(kFloat, "x");
1048 DSLFunction(kFloat, "broken").define(
1049 Declare(x, 0),
1050 If(x == 1, Return(x))
1051 );
1052 }
1053
1054 {
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001055 ExpectError error(r, "error: may not return a value from a void function\n");
1056 DSLWriter::ProgramElements().clear();
1057 DSLFunction(kVoid, "broken").define(
1058 Return(0)
1059 );
1060 }
1061
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001062 {
John Stilesb3dcbb12021-03-04 16:00:20 -05001063 ExpectError error(r, "error: function 'broken' can exit without returning a value\n");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001064 DSLWriter::ProgramElements().clear();
1065 DSLFunction(kFloat, "broken").define(
1066 );
1067 }
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001068}
1069
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001070DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLIf, r, ctxInfo) {
1071 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1072 Var a(kFloat, "a"), b(kFloat, "b");
1073 Statement x = If(a > b, a -= b);
John Stilesb4d7b582021-02-19 09:56:31 -05001074 EXPECT_EQUAL(x, "if ((a > b)) (a -= b);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001075
1076 Statement y = If(a > b, a -= b, b -= a);
John Stilesb4d7b582021-02-19 09:56:31 -05001077 EXPECT_EQUAL(y, "if ((a > b)) (a -= b); else (b -= a);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001078
1079 {
1080 ExpectError error(r, "error: expected 'bool', but found 'float'\n");
1081 If(a + b, a -= b).release();
1082 }
1083}
1084
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001085DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLReturn, r, ctxInfo) {
1086 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1087
1088 Statement x = Return();
John Stilesb4d7b582021-02-19 09:56:31 -05001089 EXPECT_EQUAL(x, "return;");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001090
1091 Statement y = Return(true);
John Stilesb4d7b582021-02-19 09:56:31 -05001092 EXPECT_EQUAL(y, "return true;");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001093}
1094
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001095DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSelect, r, ctxInfo) {
1096 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1097 Var a(kInt, "a");
1098 Expression x = Select(a > 0, 1, -1);
John Stilesb4d7b582021-02-19 09:56:31 -05001099 EXPECT_EQUAL(x, "((a > 0) ? 1 : -1)");
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001100
1101 {
1102 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001103 DSLExpression x = Select(a, 1, -1);
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001104 }
1105
1106 {
1107 ExpectError error(r, "error: ternary operator result mismatch: 'float2', 'float3'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001108 DSLExpression x = Select(a > 0, Float2(1), Float3(1));
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001109 }
1110}
1111
Ethan Nicholascfefec02021-02-09 15:22:57 -05001112DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSwitch, r, ctxInfo) {
1113 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1114
1115 Var a(kFloat, "a"), b(kInt, "b");
1116
1117 Statement x = Switch(5,
1118 Case(0, a = 0, Break()),
1119 Case(1, a = 1, Continue()),
John Stilese1d1b082021-02-23 13:44:36 -05001120 Case(2, a = 2 /*Fallthrough*/),
Ethan Nicholascfefec02021-02-09 15:22:57 -05001121 Default(Discard())
1122 );
John Stilese1d1b082021-02-23 13:44:36 -05001123 EXPECT_EQUAL(x, R"(
1124 switch (5) {
1125 case 0: (a = 0.0); break;
1126 case 1: (a = 1.0); continue;
1127 case 2: (a = 2.0);
1128 default: discard;
1129 }
1130 )");
Ethan Nicholascfefec02021-02-09 15:22:57 -05001131
John Stiles642cde22021-02-23 14:57:01 -05001132 EXPECT_EQUAL(Switch(b),
1133 "switch (b) {}");
1134
1135 EXPECT_EQUAL(Switch(b, Default(), Case(0), Case(1)),
1136 "switch (b) { default: case 0: case 1: }");
Ethan Nicholascfefec02021-02-09 15:22:57 -05001137
1138 {
John Stilese1d1b082021-02-23 13:44:36 -05001139 ExpectError error(r, "error: duplicate case value '0'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001140 DSLStatement(Switch(0, Case(0), Case(0))).release();
Ethan Nicholascfefec02021-02-09 15:22:57 -05001141 }
1142
1143 {
John Stilese1d1b082021-02-23 13:44:36 -05001144 ExpectError error(r, "error: duplicate default case\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001145 DSLStatement(Switch(0, Default(a = 0), Default(a = 1))).release();
John Stilese1d1b082021-02-23 13:44:36 -05001146 }
1147
1148 {
Ethan Nicholascfefec02021-02-09 15:22:57 -05001149 ExpectError error(r, "error: case value must be a constant integer\n");
1150 Var b(kInt);
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001151 DSLStatement(Switch(0, Case(b))).release();
Ethan Nicholascfefec02021-02-09 15:22:57 -05001152 }
1153
1154 {
1155 ExpectError error(r, "error: continue statement must be inside a loop\n");
1156 DSLFunction(kVoid, "fail").define(
1157 Switch(5,
1158 Case(0, a = 0, Break()),
1159 Case(1, a = 1, Continue()),
1160 Default(Discard())
1161 )
1162 );
1163 }
1164}
1165
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001166DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSwizzle, r, ctxInfo) {
1167 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1168 Var a(kFloat4, "a");
1169
1170 Expression e1 = a.x();
John Stilesb4d7b582021-02-19 09:56:31 -05001171 EXPECT_EQUAL(e1, "a.x");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001172
1173 Expression e2 = a.y();
John Stilesb4d7b582021-02-19 09:56:31 -05001174 EXPECT_EQUAL(e2, "a.y");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001175
1176 Expression e3 = a.z();
John Stilesb4d7b582021-02-19 09:56:31 -05001177 EXPECT_EQUAL(e3, "a.z");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001178
1179 Expression e4 = a.w();
John Stilesb4d7b582021-02-19 09:56:31 -05001180 EXPECT_EQUAL(e4, "a.w");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001181
1182 Expression e5 = a.r();
John Stilesb4d7b582021-02-19 09:56:31 -05001183 EXPECT_EQUAL(e5, "a.x");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001184
1185 Expression e6 = a.g();
John Stilesb4d7b582021-02-19 09:56:31 -05001186 EXPECT_EQUAL(e6, "a.y");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001187
1188 Expression e7 = a.b();
John Stilesb4d7b582021-02-19 09:56:31 -05001189 EXPECT_EQUAL(e7, "a.z");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001190
1191 Expression e8 = a.a();
John Stilesb4d7b582021-02-19 09:56:31 -05001192 EXPECT_EQUAL(e8, "a.w");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001193
1194 Expression e9 = Swizzle(a, R);
John Stilesb4d7b582021-02-19 09:56:31 -05001195 EXPECT_EQUAL(e9, "a.x");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001196
1197 Expression e10 = Swizzle(a, ZERO, G);
John Stiles54f00492021-02-19 11:46:10 -05001198 EXPECT_EQUAL(e10, "float2(a.y, 0.0).yx");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001199
1200 Expression e11 = Swizzle(a, B, G, G);
John Stilesb4d7b582021-02-19 09:56:31 -05001201 EXPECT_EQUAL(e11, "a.zyy");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001202
1203 Expression e12 = Swizzle(a, R, G, B, ONE);
John Stiles54f00492021-02-19 11:46:10 -05001204 EXPECT_EQUAL(e12, "float4(a.xyz, 1.0)");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001205
1206 Expression e13 = Swizzle(a, R, G, B, ONE).r();
John Stiles54f00492021-02-19 11:46:10 -05001207 EXPECT_EQUAL(e13, "float4(a.xyz, 1.0).x");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001208}
1209
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001210DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLWhile, r, ctxInfo) {
1211 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1212 Statement x = While(true, Block());
John Stilesb4d7b582021-02-19 09:56:31 -05001213 EXPECT_EQUAL(x, "for (; true;) {}");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001214
1215 Var a(kFloat, "a"), b(kFloat, "b");
1216 Statement y = While(a != b, Block(a++, --b));
John Stilesb4d7b582021-02-19 09:56:31 -05001217 EXPECT_EQUAL(y, "for (; (a != b);) { a++; --b; }");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001218
1219 {
1220 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001221 DSLStatement x = While(7, Block());
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001222 }
1223}
Ethan Nicholas04be3392021-01-26 10:07:01 -05001224
1225DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLIndex, r, ctxInfo) {
1226 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1227 Var a(Array(kInt, 5), "a"), b(kInt, "b");
John Stilesb4d7b582021-02-19 09:56:31 -05001228
1229 EXPECT_EQUAL(a[0], "a[0]");
1230 EXPECT_EQUAL(a[b], "a[b]");
Ethan Nicholas04be3392021-01-26 10:07:01 -05001231
1232 {
1233 ExpectError error(r, "error: expected 'int', but found 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001234 DSLExpression x = a[true];
Ethan Nicholas04be3392021-01-26 10:07:01 -05001235 }
1236
1237 {
1238 ExpectError error(r, "error: expected array, but found 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001239 DSLExpression x = b[0];
Ethan Nicholas04be3392021-01-26 10:07:01 -05001240 }
1241
1242 {
1243 ExpectError error(r, "error: index -1 out of range for 'int[5]'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001244 DSLExpression x = a[-1];
Ethan Nicholas04be3392021-01-26 10:07:01 -05001245 }
1246}
Ethan Nicholas30e93d52021-01-26 12:00:25 -05001247
1248DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBuiltins, r, ctxInfo) {
1249 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1250 // There is a Fract type on Mac which can conflict with our Fract builtin
1251 using SkSL::dsl::Fract;
1252 Var a(kHalf4, "a"), b(kHalf4, "b"), c(kHalf4, "c");
1253 Var h3(kHalf3, "h3");
1254 Var b4(kBool4, "b4");
John Stilesb4d7b582021-02-19 09:56:31 -05001255 EXPECT_EQUAL(Abs(a), "abs(a)");
1256 EXPECT_EQUAL(All(b4), "all(b4)");
1257 EXPECT_EQUAL(Any(b4), "any(b4)");
1258 EXPECT_EQUAL(Ceil(a), "ceil(a)");
1259 EXPECT_EQUAL(Clamp(a, 0, 1), "clamp(a, 0.0, 1.0)");
1260 EXPECT_EQUAL(Cos(a), "cos(a)");
1261 EXPECT_EQUAL(Cross(h3, h3), "cross(h3, h3)");
1262 EXPECT_EQUAL(Degrees(a), "degrees(a)");
1263 EXPECT_EQUAL(Distance(a, b), "distance(a, b)");
1264 EXPECT_EQUAL(Dot(a, b), "dot(a, b)");
1265 EXPECT_EQUAL(Equal(a, b), "equal(a, b)");
1266 EXPECT_EQUAL(Exp(a), "exp(a)");
1267 EXPECT_EQUAL(Exp2(a), "exp2(a)");
1268 EXPECT_EQUAL(Faceforward(a, b, c), "faceforward(a, b, c)");
1269 EXPECT_EQUAL(Floor(a), "floor(a)");
1270 EXPECT_EQUAL(Fract(a), "fract(a)");
1271 EXPECT_EQUAL(GreaterThan(a, b), "greaterThan(a, b)");
1272 EXPECT_EQUAL(GreaterThanEqual(a, b), "greaterThanEqual(a, b)");
1273 EXPECT_EQUAL(Inversesqrt(a), "inversesqrt(a)");
1274 EXPECT_EQUAL(LessThan(a, b), "lessThan(a, b)");
1275 EXPECT_EQUAL(LessThanEqual(a, b), "lessThanEqual(a, b)");
1276 EXPECT_EQUAL(Length(a), "length(a)");
1277 EXPECT_EQUAL(Log(a), "log(a)");
1278 EXPECT_EQUAL(Log2(a), "log2(a)");
1279 EXPECT_EQUAL(Max(a, b), "max(a, b)");
1280 EXPECT_EQUAL(Min(a, b), "min(a, b)");
1281 EXPECT_EQUAL(Mix(a, b, c), "mix(a, b, c)");
1282 EXPECT_EQUAL(Mod(a, b), "mod(a, b)");
1283 EXPECT_EQUAL(Normalize(a), "normalize(a)");
1284 EXPECT_EQUAL(NotEqual(a, b), "notEqual(a, b)");
1285 EXPECT_EQUAL(Pow(a, b), "pow(a, b)");
1286 EXPECT_EQUAL(Radians(a), "radians(a)");
1287 EXPECT_EQUAL(Reflect(a, b), "reflect(a, b)");
1288 EXPECT_EQUAL(Refract(a, b, 1), "refract(a, b, 1.0)");
1289 EXPECT_EQUAL(Saturate(a), "saturate(a)");
1290 EXPECT_EQUAL(Sign(a), "sign(a)");
1291 EXPECT_EQUAL(Sin(a), "sin(a)");
1292 EXPECT_EQUAL(Smoothstep(a, b, c), "smoothstep(a, b, c)");
1293 EXPECT_EQUAL(Sqrt(a), "sqrt(a)");
1294 EXPECT_EQUAL(Step(a, b), "step(a, b)");
1295 EXPECT_EQUAL(Tan(a), "tan(a)");
1296 EXPECT_EQUAL(Unpremul(a), "unpremul(a)");
Ethan Nicholas30e93d52021-01-26 12:00:25 -05001297
1298 // these calls all go through the normal channels, so it ought to be sufficient to prove that
1299 // one of them reports errors correctly
1300 {
1301 ExpectError error(r, "error: no match for ceil(bool)\n");
1302 Ceil(a == b).release();
1303 }
1304}
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001305
1306DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLModifiers, r, ctxInfo) {
1307 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1308
1309 Var v1(kConst_Modifier, kInt, "v1");
Ethan Nicholasbd974002021-02-22 16:20:06 -05001310 Statement d1 = Declare(v1, 0);
1311 EXPECT_EQUAL(d1, "const int v1 = 0;");
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001312
1313 // Most modifiers require an appropriate context to be legal. We can't yet give them that
1314 // context, so we can't as yet Declare() variables with these modifiers.
1315 // TODO: better tests when able
1316 Var v2(kIn_Modifier, kInt, "v2");
1317 REPORTER_ASSERT(r, DSLWriter::Var(v2).modifiers().fFlags == SkSL::Modifiers::kIn_Flag);
1318
1319 Var v3(kOut_Modifier, kInt, "v3");
1320 REPORTER_ASSERT(r, DSLWriter::Var(v3).modifiers().fFlags == SkSL::Modifiers::kOut_Flag);
1321
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001322 Var v4(kFlat_Modifier, kInt, "v4");
1323 REPORTER_ASSERT(r, DSLWriter::Var(v4).modifiers().fFlags == SkSL::Modifiers::kFlat_Flag);
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001324
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001325 Var v5(kNoPerspective_Modifier, kInt, "v5");
1326 REPORTER_ASSERT(r, DSLWriter::Var(v5).modifiers().fFlags ==
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001327 SkSL::Modifiers::kNoPerspective_Flag);
1328
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001329 Var v6(kIn_Modifier | kOut_Modifier, kInt, "v6");
1330 REPORTER_ASSERT(r, DSLWriter::Var(v6).modifiers().fFlags ==
1331 (SkSL::Modifiers::kIn_Flag | SkSL::Modifiers::kOut_Flag));
1332
1333 Var v7(kInOut_Modifier, kInt, "v7");
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001334 REPORTER_ASSERT(r, DSLWriter::Var(v7).modifiers().fFlags ==
1335 (SkSL::Modifiers::kIn_Flag | SkSL::Modifiers::kOut_Flag));
1336
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001337 Var v8(kUniform_Modifier, kInt, "v8");
1338 REPORTER_ASSERT(r, DSLWriter::Var(v8).modifiers().fFlags == SkSL::Modifiers::kUniform_Flag);
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001339}
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001340
1341DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLStruct, r, ctxInfo) {
1342 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1343
1344 DSLType simpleStruct = Struct("SimpleStruct",
1345 Field(kFloat, "x"),
1346 Field(kBool, "b"),
1347 Field(Array(kFloat, 3), "a")
1348 );
1349 DSLVar result(simpleStruct, "result");
1350 DSLFunction(simpleStruct, "returnStruct").define(
1351 Declare(result),
1352 result.field("x") = 123,
1353 result.field("b") = result.field("x") > 0,
1354 result.field("a")[0] = result.field("x"),
1355 Return(result)
1356 );
1357 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 2);
John Stilesb4d7b582021-02-19 09:56:31 -05001358 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1359 "struct SimpleStruct { float x; bool b; float[3] a; };");
1360 EXPECT_EQUAL(*DSLWriter::ProgramElements()[1],
1361 "SimpleStruct returnStruct() { SimpleStruct result; (result.x = 123.0);"
1362 "(result.b = (result.x > 0.0)); (result.a[0] = result.x); return result; }");
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001363
1364 DSLWriter::ProgramElements().clear();
1365 Struct("NestedStruct",
1366 Field(kInt, "x"),
1367 Field(simpleStruct, "simple")
1368 );
1369 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
John Stilesb4d7b582021-02-19 09:56:31 -05001370 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1371 "struct NestedStruct { int x; SimpleStruct simple; };");
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001372}