blob: 77844df7eeba60017db7b8770d470e8ed7876353 [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),
139 "float3(float2(0.0, 1.0), -2.0)");
140 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)),
145 "float4(float2(0.0, 1.0), float2(2.0, 3.0))");
146 EXPECT_EQUAL(Float4(0, 1, Float2(2, 3)),
147 "float4(0.0, 1.0, float2(2.0, 3.0))");
148 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
184 Expression e3 = Half2(0);
John Stilesb4d7b582021-02-19 09:56:31 -0500185 EXPECT_EQUAL(e3, "half2(0.0)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500186
187 Expression e4 = Half2(-0.5, 1);
John Stilesb4d7b582021-02-19 09:56:31 -0500188 EXPECT_EQUAL(e4, "half2(-0.5, 1.0)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500189
190 Expression e5 = Half3(0.75);
John Stilesb4d7b582021-02-19 09:56:31 -0500191 EXPECT_EQUAL(e5, "half3(0.75)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500192
193 Expression e6 = Half3(Half2(0, 1), -2);
John Stilesb4d7b582021-02-19 09:56:31 -0500194 EXPECT_EQUAL(e6, "half3(half2(0.0, 1.0), -2.0)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500195
196 Expression e7 = Half3(0, 1, 2);
John Stilesb4d7b582021-02-19 09:56:31 -0500197 EXPECT_EQUAL(e7, "half3(0.0, 1.0, 2.0)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500198
199 Expression e8 = Half4(0);
John Stilesb4d7b582021-02-19 09:56:31 -0500200 EXPECT_EQUAL(e8, "half4(0.0)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500201
202 Expression e9 = Half4(Half2(0, 1), Half2(2, 3));
John Stilesb4d7b582021-02-19 09:56:31 -0500203 EXPECT_EQUAL(e9, "half4(half2(0.0, 1.0), half2(2.0, 3.0))");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500204
205 Expression e10 = Half4(0, 1, Half2(2, 3));
John Stilesb4d7b582021-02-19 09:56:31 -0500206 EXPECT_EQUAL(e10, "half4(0.0, 1.0, half2(2.0, 3.0))");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500207
208 Expression e11 = Half4(0, 1, 2, 3);
John Stilesb4d7b582021-02-19 09:56:31 -0500209 EXPECT_EQUAL(e11, "half4(0.0, 1.0, 2.0, 3.0)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500210
211 {
212 ExpectError error(r, "error: floating point value is infinite\n");
213 Half(std::numeric_limits<float>::infinity()).release();
214 }
215
216 {
217 ExpectError error(r, "error: floating point value is NaN\n");
218 Half(std::numeric_limits<float>::quiet_NaN()).release();
219 }
220
221 {
222 ExpectError error(r, "error: invalid arguments to 'half2' constructor (expected 2 scalars,"
223 " but found 4)\n");
224 Half2(Half4(1)).release();
225 }
226
227 {
228 ExpectError error(r, "error: invalid arguments to 'half4' constructor (expected 4 scalars,"
229 " but found 3)\n");
230 Half4(Half3(1)).release();
231 }
232}
233
234DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLInt, r, ctxInfo) {
235 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
236 Expression e1 = Int(std::numeric_limits<int32_t>::max());
John Stilesb4d7b582021-02-19 09:56:31 -0500237 EXPECT_EQUAL(e1, "2147483647");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500238
239 Expression e2 = Int2(std::numeric_limits<int32_t>::min());
John Stilesb4d7b582021-02-19 09:56:31 -0500240 EXPECT_EQUAL(e2, "int2(-2147483648)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500241
242 Expression e3 = Int2(0, 1);
John Stilesb4d7b582021-02-19 09:56:31 -0500243 EXPECT_EQUAL(e3, "int2(0, 1)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500244
245 Expression e4 = Int3(0);
John Stilesb4d7b582021-02-19 09:56:31 -0500246 EXPECT_EQUAL(e4, "int3(0)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500247
248 Expression e5 = Int3(Int2(0, 1), -2);
John Stilesb4d7b582021-02-19 09:56:31 -0500249 EXPECT_EQUAL(e5, "int3(int2(0, 1), -2)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500250
251 Expression e6 = Int3(0, 1, 2);
John Stilesb4d7b582021-02-19 09:56:31 -0500252 EXPECT_EQUAL(e6, "int3(0, 1, 2)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500253
254 Expression e7 = Int4(0);
John Stilesb4d7b582021-02-19 09:56:31 -0500255 EXPECT_EQUAL(e7, "int4(0)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500256
257 Expression e8 = Int4(Int2(0, 1), Int2(2, 3));
John Stilesb4d7b582021-02-19 09:56:31 -0500258 EXPECT_EQUAL(e8, "int4(int2(0, 1), int2(2, 3))");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500259
260 Expression e9 = Int4(0, 1, Int2(2, 3));
John Stilesb4d7b582021-02-19 09:56:31 -0500261 EXPECT_EQUAL(e9, "int4(0, 1, int2(2, 3))");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500262
263 Expression e10 = Int4(0, 1, 2, 3);
John Stilesb4d7b582021-02-19 09:56:31 -0500264 EXPECT_EQUAL(e10, "int4(0, 1, 2, 3)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500265
266 {
267 ExpectError error(r, "error: invalid arguments to 'int2' constructor (expected 2 scalars,"
268 " but found 4)\n");
269 Int2(Int4(1)).release();
270 }
271
272 {
273 ExpectError error(r, "error: invalid arguments to 'int4' constructor (expected 4 scalars,"
274 " but found 3)\n");
275 Int4(Int3(1)).release();
276 }
277}
278
279DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLShort, r, ctxInfo) {
280 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
281 Expression e1 = Short(std::numeric_limits<int16_t>::max());
John Stilesb4d7b582021-02-19 09:56:31 -0500282 EXPECT_EQUAL(e1, "32767");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500283
284 Expression e2 = Short2(std::numeric_limits<int16_t>::min());
John Stilesb4d7b582021-02-19 09:56:31 -0500285 EXPECT_EQUAL(e2, "short2(-32768)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500286
287 Expression e3 = Short2(0, 1);
John Stilesb4d7b582021-02-19 09:56:31 -0500288 EXPECT_EQUAL(e3, "short2(0, 1)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500289
290 Expression e4 = Short3(0);
John Stilesb4d7b582021-02-19 09:56:31 -0500291 EXPECT_EQUAL(e4, "short3(0)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500292
293 Expression e5 = Short3(Short2(0, 1), -2);
John Stilesb4d7b582021-02-19 09:56:31 -0500294 EXPECT_EQUAL(e5, "short3(short2(0, 1), -2)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500295
296 Expression e6 = Short3(0, 1, 2);
John Stilesb4d7b582021-02-19 09:56:31 -0500297 EXPECT_EQUAL(e6, "short3(0, 1, 2)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500298
299 Expression e7 = Short4(0);
John Stilesb4d7b582021-02-19 09:56:31 -0500300 EXPECT_EQUAL(e7, "short4(0)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500301
302 Expression e8 = Short4(Short2(0, 1), Short2(2, 3));
John Stilesb4d7b582021-02-19 09:56:31 -0500303 EXPECT_EQUAL(e8, "short4(short2(0, 1), short2(2, 3))");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500304
305 Expression e9 = Short4(0, 1, Short2(2, 3));
John Stilesb4d7b582021-02-19 09:56:31 -0500306 EXPECT_EQUAL(e9, "short4(0, 1, short2(2, 3))");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500307
308 Expression e10 = Short4(0, 1, 2, 3);
John Stilesb4d7b582021-02-19 09:56:31 -0500309 EXPECT_EQUAL(e10, "short4(0, 1, 2, 3)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500310
311 {
312 ExpectError error(r, "error: invalid arguments to 'short2' constructor (expected 2 scalars,"
313 " but found 4)\n");
314 Short2(Short4(1)).release();
315 }
316
317 {
318 ExpectError error(r, "error: invalid arguments to 'short4' constructor (expected 4 scalars,"
319 " but found 3)\n");
320 Short4(Short3(1)).release();
321 }
322}
323
324DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBool, r, ctxInfo) {
325 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
326 Expression e1 = Bool2(false);
John Stilesb4d7b582021-02-19 09:56:31 -0500327 EXPECT_EQUAL(e1, "bool2(false)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500328
329 Expression e2 = Bool2(false, true);
John Stilesb4d7b582021-02-19 09:56:31 -0500330 EXPECT_EQUAL(e2, "bool2(false, true)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500331
332 Expression e3 = Bool3(false);
John Stilesb4d7b582021-02-19 09:56:31 -0500333 EXPECT_EQUAL(e3, "bool3(false)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500334
335 Expression e4 = Bool3(Bool2(false, true), false);
John Stilesb4d7b582021-02-19 09:56:31 -0500336 EXPECT_EQUAL(e4, "bool3(bool2(false, true), false)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500337
338 Expression e5 = Bool3(false, true, false);
John Stilesb4d7b582021-02-19 09:56:31 -0500339 EXPECT_EQUAL(e5, "bool3(false, true, false)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500340
341 Expression e6 = Bool4(false);
John Stilesb4d7b582021-02-19 09:56:31 -0500342 EXPECT_EQUAL(e6, "bool4(false)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500343
344 Expression e7 = Bool4(Bool2(false, true), Bool2(false, true));
John Stilesb4d7b582021-02-19 09:56:31 -0500345 EXPECT_EQUAL(e7, "bool4(bool2(false, true), "
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500346 "bool2(false, true))");
347
348 Expression e8 = Bool4(false, true, Bool2(false, true));
John Stilesb4d7b582021-02-19 09:56:31 -0500349 EXPECT_EQUAL(e8, "bool4(false, true, bool2(false, true))");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500350
351 Expression e9 = Bool4(false, true, false, true);
John Stilesb4d7b582021-02-19 09:56:31 -0500352 EXPECT_EQUAL(e9, "bool4(false, true, false, true)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500353
354 {
355 ExpectError error(r, "error: invalid arguments to 'bool2' constructor (expected 2 scalars,"
356 " but found 4)\n");
357 Bool2(Bool4(true)).release();
358 }
359
360 {
361 ExpectError error(r, "error: invalid arguments to 'bool4' constructor (expected 4 scalars,"
362 " but found 3)\n");
363 Bool4(Bool3(true)).release();
364 }
365}
Ethan Nicholas92969f22021-01-13 10:38:59 -0500366
367DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLPlus, r, ctxInfo) {
368 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
369 Var a(kFloat, "a"), b(kFloat, "b");
370 Expression e1 = a + b;
John Stilesb4d7b582021-02-19 09:56:31 -0500371 EXPECT_EQUAL(e1, "(a + b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500372
373 Expression e2 = a + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500374 EXPECT_EQUAL(e2, "(a + 1.0)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500375
376 Expression e3 = 0.5 + a + -99;
John Stilesb4d7b582021-02-19 09:56:31 -0500377 EXPECT_EQUAL(e3, "((0.5 + a) + -99.0)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500378
379 Expression e4 = a += b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500380 EXPECT_EQUAL(e4, "(a += (b + 1.0))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500381
382 {
383 ExpectError error(r, "error: type mismatch: '+' cannot operate on 'bool2', 'float'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500384 DSLExpression((Bool2(true) + a)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500385 }
386
387 {
388 ExpectError error(r, "error: type mismatch: '+=' cannot operate on 'float', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500389 DSLExpression((a += Bool2(true))).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500390 }
391
392 {
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500393 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500394 DSLExpression((1.0 += a)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500395 }
396}
397
398DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLMinus, r, ctxInfo) {
399 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
400 Var a(kInt, "a"), b(kInt, "b");
401 Expression e1 = a - b;
John Stilesb4d7b582021-02-19 09:56:31 -0500402 EXPECT_EQUAL(e1, "(a - b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500403
404 Expression e2 = a - 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500405 EXPECT_EQUAL(e2, "(a - 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500406
407 Expression e3 = 2 - a - b;
John Stilesb4d7b582021-02-19 09:56:31 -0500408 EXPECT_EQUAL(e3, "((2 - a) - b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500409
410 Expression e4 = a -= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500411 EXPECT_EQUAL(e4, "(a -= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500412
413 {
414 ExpectError error(r, "error: type mismatch: '-' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500415 DSLExpression(Bool2(true) - a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500416 }
417
418 {
419 ExpectError error(r, "error: type mismatch: '-=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500420 DSLExpression(a -= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500421 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500422
423 {
424 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500425 DSLExpression(1.0 -= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500426 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500427}
428
429DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLMultiply, r, ctxInfo) {
430 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
431 Var a(kFloat, "a"), b(kFloat, "b");
432 Expression e1 = a * b;
John Stilesb4d7b582021-02-19 09:56:31 -0500433 EXPECT_EQUAL(e1, "(a * b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500434
435 Expression e2 = a * 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500436 EXPECT_EQUAL(e2, "(a * 1.0)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500437
438 Expression e3 = 0.5 * a * -99;
John Stilesb4d7b582021-02-19 09:56:31 -0500439 EXPECT_EQUAL(e3, "((0.5 * a) * -99.0)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500440
441 Expression e4 = a *= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500442 EXPECT_EQUAL(e4, "(a *= (b + 1.0))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500443
444 {
445 ExpectError error(r, "error: type mismatch: '*' cannot operate on 'bool2', 'float'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500446 DSLExpression(Bool2(true) * a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500447 }
448
449 {
450 ExpectError error(r, "error: type mismatch: '*=' cannot operate on 'float', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500451 DSLExpression(a *= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500452 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500453
454 {
455 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500456 DSLExpression(1.0 *= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500457 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500458}
459
460DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDivide, r, ctxInfo) {
461 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
462 Var a(kFloat, "a"), b(kFloat, "b");
463 Expression e1 = a / b;
John Stilesb4d7b582021-02-19 09:56:31 -0500464 EXPECT_EQUAL(e1, "(a / b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500465
466 Expression e2 = a / 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500467 EXPECT_EQUAL(e2, "(a / 1.0)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500468
469 Expression e3 = 0.5 / a / -99;
John Stilesb4d7b582021-02-19 09:56:31 -0500470 EXPECT_EQUAL(e3, "((0.5 / a) / -99.0)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500471
472 Expression e4 = b / (a - 1);
John Stilesb4d7b582021-02-19 09:56:31 -0500473 EXPECT_EQUAL(e4, "(b / (a - 1.0))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500474
475 Expression e5 = a /= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500476 EXPECT_EQUAL(e5, "(a /= (b + 1.0))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500477
478 {
479 ExpectError error(r, "error: type mismatch: '/' cannot operate on 'bool2', 'float'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500480 DSLExpression(Bool2(true) / a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500481 }
482
483 {
484 ExpectError error(r, "error: type mismatch: '/=' cannot operate on 'float', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500485 DSLExpression(a /= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500486 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500487
488 {
489 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500490 DSLExpression(1.0 /= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500491 }
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500492
493 {
494 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500495 DSLExpression(a /= 0).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500496 }
497
498 {
499 Var c(kFloat2, "c");
500 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500501 DSLExpression(c /= Float2(Float(0), 1)).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500502 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500503}
504
505DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLMod, r, ctxInfo) {
506 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
507 Var a(kInt, "a"), b(kInt, "b");
508 Expression e1 = a % b;
John Stilesb4d7b582021-02-19 09:56:31 -0500509 EXPECT_EQUAL(e1, "(a % b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500510
511 Expression e2 = a % 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500512 EXPECT_EQUAL(e2, "(a % 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500513
514 Expression e3 = 10 % a % -99;
John Stilesb4d7b582021-02-19 09:56:31 -0500515 EXPECT_EQUAL(e3, "((10 % a) % -99)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500516
517 Expression e4 = a %= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500518 EXPECT_EQUAL(e4, "(a %= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500519
520 {
521 ExpectError error(r, "error: type mismatch: '%' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500522 DSLExpression(Bool2(true) % a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500523 }
524
525 {
526 ExpectError error(r, "error: type mismatch: '%=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500527 DSLExpression(a %= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500528 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500529
530 {
531 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500532 DSLExpression(1 %= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500533 }
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500534
535 {
536 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500537 DSLExpression(a %= 0).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500538 }
539
540 {
541 Var c(kInt2, "c");
542 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500543 DSLExpression(c %= Int2(Int(0), 1)).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500544 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500545}
546
547DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLShl, r, ctxInfo) {
548 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
549 Var a(kInt, "a"), b(kInt, "b");
550 Expression e1 = a << b;
John Stilesb4d7b582021-02-19 09:56:31 -0500551 EXPECT_EQUAL(e1, "(a << b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500552
553 Expression e2 = a << 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500554 EXPECT_EQUAL(e2, "(a << 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500555
556 Expression e3 = 1 << a << 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500557 EXPECT_EQUAL(e3, "((1 << a) << 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500558
559 Expression e4 = a <<= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500560 EXPECT_EQUAL(e4, "(a <<= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500561
562 {
563 ExpectError error(r, "error: type mismatch: '<<' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500564 DSLExpression(Bool2(true) << a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500565 }
566
567 {
568 ExpectError error(r, "error: type mismatch: '<<=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500569 DSLExpression(a <<= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500570 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500571
572 {
573 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500574 DSLExpression(1 <<= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500575 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500576}
577
578DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLShr, r, ctxInfo) {
579 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
580 Var a(kInt, "a"), b(kInt, "b");
581 Expression e1 = a >> b;
John Stilesb4d7b582021-02-19 09:56:31 -0500582 EXPECT_EQUAL(e1, "(a >> b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500583
584 Expression e2 = a >> 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500585 EXPECT_EQUAL(e2, "(a >> 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500586
587 Expression e3 = 1 >> a >> 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500588 EXPECT_EQUAL(e3, "((1 >> a) >> 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500589
590 Expression e4 = a >>= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500591 EXPECT_EQUAL(e4, "(a >>= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500592
593 {
594 ExpectError error(r, "error: type mismatch: '>>' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500595 DSLExpression(Bool2(true) >> a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500596 }
597
598 {
599 ExpectError error(r, "error: type mismatch: '>>=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500600 DSLExpression(a >>= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500601 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500602
603 {
604 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500605 DSLExpression(1 >>= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500606 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500607}
608
609DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseAnd, r, ctxInfo) {
610 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
611 Var a(kInt, "a"), b(kInt, "b");
612 Expression e1 = a & b;
John Stilesb4d7b582021-02-19 09:56:31 -0500613 EXPECT_EQUAL(e1, "(a & b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500614
615 Expression e2 = a & 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500616 EXPECT_EQUAL(e2, "(a & 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500617
618 Expression e3 = 1 & a & 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500619 EXPECT_EQUAL(e3, "((1 & a) & 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500620
621 Expression e4 = a &= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500622 EXPECT_EQUAL(e4, "(a &= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500623
624 {
625 ExpectError error(r, "error: type mismatch: '&' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500626 DSLExpression(Bool2(true) & a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500627 }
628
629 {
630 ExpectError error(r, "error: type mismatch: '&=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500631 DSLExpression(a &= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500632 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500633
634 {
635 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500636 DSLExpression(1 &= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500637 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500638}
639
640DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseOr, r, ctxInfo) {
641 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
642 Var a(kInt, "a"), b(kInt, "b");
643 Expression e1 = a | b;
John Stilesb4d7b582021-02-19 09:56:31 -0500644 EXPECT_EQUAL(e1, "(a | b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500645
646 Expression e2 = a | 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500647 EXPECT_EQUAL(e2, "(a | 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500648
649 Expression e3 = 1 | a | 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500650 EXPECT_EQUAL(e3, "((1 | a) | 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500651
652 Expression e4 = a |= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500653 EXPECT_EQUAL(e4, "(a |= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500654
655 {
656 ExpectError error(r, "error: type mismatch: '|' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500657 DSLExpression(Bool2(true) | a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500658 }
659
660 {
661 ExpectError error(r, "error: type mismatch: '|=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500662 DSLExpression(a |= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500663 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500664
665 {
666 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500667 DSLExpression(1 |= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500668 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500669}
670
671DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseXor, r, ctxInfo) {
672 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
673 Var a(kInt, "a"), b(kInt, "b");
674 Expression e1 = a ^ b;
John Stilesb4d7b582021-02-19 09:56:31 -0500675 EXPECT_EQUAL(e1, "(a ^ b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500676
677 Expression e2 = a ^ 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500678 EXPECT_EQUAL(e2, "(a ^ 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500679
680 Expression e3 = 1 ^ a ^ 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500681 EXPECT_EQUAL(e3, "((1 ^ a) ^ 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500682
683 Expression e4 = a ^= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500684 EXPECT_EQUAL(e4, "(a ^= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500685
686 {
687 ExpectError error(r, "error: type mismatch: '^' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500688 DSLExpression(Bool2(true) ^ a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500689 }
690
691 {
692 ExpectError error(r, "error: type mismatch: '^=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500693 DSLExpression(a ^= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500694 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500695
696 {
697 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500698 DSLExpression(1 ^= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500699 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500700}
701
702DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLogicalAnd, r, ctxInfo) {
703 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
704 Var a(kBool, "a"), b(kBool, "b");
705 Expression e1 = a && b;
John Stilesb4d7b582021-02-19 09:56:31 -0500706 EXPECT_EQUAL(e1, "(a && b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500707
708 Expression e2 = a && true && b;
John Stilesb4d7b582021-02-19 09:56:31 -0500709 EXPECT_EQUAL(e2, "(a && b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500710
711 Expression e3 = a && false && b;
John Stilesb4d7b582021-02-19 09:56:31 -0500712 EXPECT_EQUAL(e3, "false");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500713
714 {
715 ExpectError error(r, "error: type mismatch: '&&' cannot operate on 'bool', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500716 DSLExpression(a && 5).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500717 }
718}
719
720DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLogicalOr, r, ctxInfo) {
721 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
722 Var a(kBool, "a"), b(kBool, "b");
723 Expression e1 = a || b;
John Stilesb4d7b582021-02-19 09:56:31 -0500724 EXPECT_EQUAL(e1, "(a || b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500725
726 Expression e2 = a || true || b;
John Stilesb4d7b582021-02-19 09:56:31 -0500727 EXPECT_EQUAL(e2, "true");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500728
729 Expression e3 = a || false || b;
John Stilesb4d7b582021-02-19 09:56:31 -0500730 EXPECT_EQUAL(e3, "(a || b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500731
732 {
733 ExpectError error(r, "error: type mismatch: '||' cannot operate on 'bool', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500734 DSLExpression(a || 5).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500735 }
736}
737
738DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLComma, r, ctxInfo) {
739 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
740 Var a(kInt, "a"), b(kInt, "b");
741 Expression e1 = (a += b, b);
John Stilesb4d7b582021-02-19 09:56:31 -0500742 EXPECT_EQUAL(e1, "((a += b) , b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500743
744 Expression e2 = (a += b, b += b, Int2(a));
John Stilesb4d7b582021-02-19 09:56:31 -0500745 EXPECT_EQUAL(e2, "(((a += b) , (b += b)) , int2(a))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500746}
747
748DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLEqual, r, ctxInfo) {
749 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
750 Var a(kInt, "a"), b(kInt, "b");
751 Expression e1 = a == b;
John Stilesb4d7b582021-02-19 09:56:31 -0500752 EXPECT_EQUAL(e1, "(a == b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500753
754 Expression e2 = a == 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500755 EXPECT_EQUAL(e2, "(a == 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500756
757 {
758 ExpectError error(r, "error: type mismatch: '==' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500759 DSLExpression(a == Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500760 }
761}
762
763DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLNotEqual, r, ctxInfo) {
764 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
765 Var a(kInt, "a"), b(kInt, "b");
766 Expression e1 = a != b;
John Stilesb4d7b582021-02-19 09:56:31 -0500767 EXPECT_EQUAL(e1, "(a != b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500768
769 Expression e2 = a != 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500770 EXPECT_EQUAL(e2, "(a != 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500771
772 {
773 ExpectError error(r, "error: type mismatch: '!=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500774 DSLExpression(a != Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500775 }
776}
777
778DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLGreaterThan, r, ctxInfo) {
779 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
780 Var a(kInt, "a"), b(kInt, "b");
781 Expression e1 = a > b;
John Stilesb4d7b582021-02-19 09:56:31 -0500782 EXPECT_EQUAL(e1, "(a > b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500783
784 Expression e2 = a > 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500785 EXPECT_EQUAL(e2, "(a > 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500786
787 {
788 ExpectError error(r, "error: type mismatch: '>' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500789 DSLExpression(a > Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500790 }
791}
792
793DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLGreaterThanOrEqual, r, ctxInfo) {
794 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
795 Var a(kInt, "a"), b(kInt, "b");
796 Expression e1 = a >= b;
John Stilesb4d7b582021-02-19 09:56:31 -0500797 EXPECT_EQUAL(e1, "(a >= b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500798
799 Expression e2 = a >= 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500800 EXPECT_EQUAL(e2, "(a >= 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500801
802 {
803 ExpectError error(r, "error: type mismatch: '>=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500804 DSLExpression(a >= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500805 }
806}
807
808DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLessThan, r, ctxInfo) {
809 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
810 Var a(kInt, "a"), b(kInt, "b");
811 Expression e1 = a < b;
John Stilesb4d7b582021-02-19 09:56:31 -0500812 EXPECT_EQUAL(e1, "(a < b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500813
814 Expression e2 = a < 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500815 EXPECT_EQUAL(e2, "(a < 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500816
817 {
818 ExpectError error(r, "error: type mismatch: '<' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500819 DSLExpression(a < Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500820 }
821}
822
823DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLessThanOrEqual, r, ctxInfo) {
824 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
825 Var a(kInt, "a"), b(kInt, "b");
826 Expression e1 = a <= b;
John Stilesb4d7b582021-02-19 09:56:31 -0500827 EXPECT_EQUAL(e1, "(a <= b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500828
829 Expression e2 = a <= 5;
John Stilesb4d7b582021-02-19 09:56:31 -0500830 EXPECT_EQUAL(e2, "(a <= 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500831
832 {
833 ExpectError error(r, "error: type mismatch: '<=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500834 DSLExpression(a <= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500835 }
836}
837
838DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLogicalNot, r, ctxInfo) {
839 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
840 Var a(kInt, "a"), b(kInt, "b");
841 Expression e1 = !(a <= b);
John Stilesb4d7b582021-02-19 09:56:31 -0500842 EXPECT_EQUAL(e1, "!(a <= b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500843
844 {
845 ExpectError error(r, "error: '!' cannot operate on 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500846 DSLExpression(!a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500847 }
848}
849
850DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseNot, r, ctxInfo) {
851 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
852 Var a(kInt, "a"), b(kBool, "b");
853 Expression e1 = ~a;
John Stilesb4d7b582021-02-19 09:56:31 -0500854 EXPECT_EQUAL(e1, "~a");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500855
856 {
857 ExpectError error(r, "error: '~' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500858 DSLExpression(~b).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500859 }
860}
861
862DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLIncrement, r, ctxInfo) {
863 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
864 Var a(kInt, "a"), b(kBool, "b");
865 Expression e1 = ++a;
John Stilesb4d7b582021-02-19 09:56:31 -0500866 EXPECT_EQUAL(e1, "++a");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500867
868 Expression e2 = a++;
John Stilesb4d7b582021-02-19 09:56:31 -0500869 EXPECT_EQUAL(e2, "a++");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500870
871 {
872 ExpectError error(r, "error: '++' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500873 DSLExpression(++b).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500874 }
875
876 {
877 ExpectError error(r, "error: '++' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500878 DSLExpression(b++).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500879 }
880
881 {
882 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500883 DSLExpression(++(a + 1)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500884 }
885
886 {
887 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500888 DSLExpression((a + 1)++).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500889 }
890}
891
892DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDecrement, r, ctxInfo) {
893 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
894 Var a(kInt, "a"), b(kBool, "b");
895 Expression e1 = --a;
John Stilesb4d7b582021-02-19 09:56:31 -0500896 EXPECT_EQUAL(e1, "--a");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500897
898 Expression e2 = a--;
John Stilesb4d7b582021-02-19 09:56:31 -0500899 EXPECT_EQUAL(e2, "a--");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500900
901 {
902 ExpectError error(r, "error: '--' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500903 DSLExpression(--b).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500904 }
905
906 {
907 ExpectError error(r, "error: '--' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500908 DSLExpression(b--).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500909 }
910
911 {
912 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500913 DSLExpression(--(a + 1)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500914 }
915
916 {
917 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500918 DSLExpression((a + 1)--).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500919 }
920}
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500921
922DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBlock, r, ctxInfo) {
923 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
924 Statement x = Block();
John Stilesb4d7b582021-02-19 09:56:31 -0500925 EXPECT_EQUAL(x, "{ }");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500926 Var a(kInt, "a"), b(kInt, "b");
927 Statement y = Block(Declare(a, 1), Declare(b, 2), a = b);
John Stilesb4d7b582021-02-19 09:56:31 -0500928 EXPECT_EQUAL(y, "{ int a = 1; int b = 2; (a = b); }");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500929}
930
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500931DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBreak, r, ctxInfo) {
932 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
933 Var i(kInt, "i");
934 DSLFunction(kVoid, "success").define(
935 For(Declare(i, 0), i < 10, ++i, Block(
936 If(i > 5, Break())
937 ))
938 );
939 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
John Stilesb4d7b582021-02-19 09:56:31 -0500940 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
941 "void success() { for (int i = 0; (i < 10); ++i) { if ((i > 5)) break; } }");
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500942
943 {
944 ExpectError error(r, "error: break statement must be inside a loop or switch\n");
945 DSLFunction(kVoid, "fail").define(
946 Break()
947 );
948 }
949}
950
951DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLContinue, r, ctxInfo) {
952 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
953 Var i(kInt, "i");
954 DSLFunction(kVoid, "success").define(
955 For(Declare(i, 0), i < 10, ++i, Block(
956 If(i < 5, Continue())
957 ))
958 );
959 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
John Stilesb4d7b582021-02-19 09:56:31 -0500960 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
961 "void success() { for (int i = 0; (i < 10); ++i) { if ((i < 5)) continue; } }");
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500962
963 {
964 ExpectError error(r, "error: continue statement must be inside a loop\n");
965 DSLFunction(kVoid, "fail").define(
966 Continue()
967 );
968 }
969}
970
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500971DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDeclare, r, ctxInfo) {
972 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
973 Var a(kHalf4, "a"), b(kHalf4, "b");
974 Statement x = Declare(a);
John Stilesb4d7b582021-02-19 09:56:31 -0500975 EXPECT_EQUAL(x, "half4 a;");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500976 Statement y = Declare(b, Half4(1));
John Stilesb4d7b582021-02-19 09:56:31 -0500977 EXPECT_EQUAL(y, "half4 b = half4(1.0);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500978
979 {
980 Var c(kHalf4, "c");
981 ExpectError error(r, "error: expected 'half4', but found 'int'\n");
982 Declare(c, 1).release();
983 }
984}
985
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500986DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDiscard, r, ctxInfo) {
987 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
988 Statement x = If(Sqrt(1) > 0, Discard());
John Stilesb4d7b582021-02-19 09:56:31 -0500989 EXPECT_EQUAL(x, "if ((sqrt(1.0) > 0.0)) discard;");
Ethan Nicholasdaceb792021-02-05 14:22:32 -0500990}
991
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500992DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDo, r, ctxInfo) {
993 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
994 Statement x = Do(Block(), true);
John Stilesb4d7b582021-02-19 09:56:31 -0500995 EXPECT_EQUAL(x, "do {} while (true);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500996
997 Var a(kFloat, "a"), b(kFloat, "b");
998 Statement y = Do(Block(a++, --b), a != b);
John Stilesb4d7b582021-02-19 09:56:31 -0500999 EXPECT_EQUAL(y, "do { a++; --b; } while ((a != b));");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001000
1001 {
1002 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
1003 Do(Block(), 7).release();
1004 }
1005}
1006
1007DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLFor, r, ctxInfo) {
1008 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1009 Statement x = For(Statement(), Expression(), Expression(), Block());
John Stilesb4d7b582021-02-19 09:56:31 -05001010 EXPECT_EQUAL(x, "for (;;) {}");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001011
1012 Var i(kInt, "i");
1013 Statement y = For(Declare(i, 0), i < 10, ++i, i += 5);
John Stilesb4d7b582021-02-19 09:56:31 -05001014 EXPECT_EQUAL(y, "for (int i = 0; (i < 10); ++i) (i += 5);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001015
1016 {
1017 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
1018 For(i = 0, i + 10, ++i, i += 5).release();
1019 }
1020}
1021
Ethan Nicholase2c05042021-02-03 10:27:22 -05001022DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLFunction, r, ctxInfo) {
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001023 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1024 DSLWriter::ProgramElements().clear();
1025 Var coords(kHalf2, "coords");
1026 DSLFunction(kVoid, "main", coords).define(
1027 sk_FragColor() = Half4(coords, 0, 1)
1028 );
1029 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
John Stilesb4d7b582021-02-19 09:56:31 -05001030 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1031 "void main(half2 coords) { (sk_FragColor = half4(coords, 0.0, 1.0)); }");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001032
Ethan Nicholas63f75fc2021-02-23 12:05:49 -05001033 {
1034 DSLWriter::ProgramElements().clear();
1035 Var x(kFloat, "x");
1036 DSLFunction sqr(kFloat, "sqr", x);
1037 sqr.define(
1038 Return(x * x)
1039 );
1040 EXPECT_EQUAL(sqr(sk_FragCoord().x()), "sqr(sk_FragCoord.x)");
1041 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
1042 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0], "float sqr(float x) { return (x * x); }");
1043 }
1044
1045 {
1046 DSLWriter::ProgramElements().clear();
1047 Var x(kFloat2, "x");
1048 Var y(kFloat2, "y");
1049 DSLFunction dot(kFloat2, "dot", x, y);
1050 dot.define(
1051 Return(x * x + y * y)
1052 );
1053 EXPECT_EQUAL(dot(Float2(1.0f, 2.0f), Float2(3.0f, 4.0f)),
1054 "dot(float2(1.0, 2.0), float2(3.0, 4.0))");
1055 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
1056 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1057 "float2 dot(float2 x, float2 y) { return ((x * x) + (y * y)); }");
1058 }
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001059
1060 {
1061 ExpectError error(r, "error: expected 'float', but found 'bool'\n");
1062 DSLWriter::ProgramElements().clear();
1063 DSLFunction(kFloat, "broken").define(
1064 Return(true)
1065 );
1066 }
1067
1068 {
John Stilesb3dcbb12021-03-04 16:00:20 -05001069 ExpectError error(r, "error: expected function to return 'float'\n");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001070 DSLWriter::ProgramElements().clear();
1071 DSLFunction(kFloat, "broken").define(
1072 Return()
1073 );
1074 }
1075
1076 {
John Stilesb3dcbb12021-03-04 16:00:20 -05001077 ExpectError error(r, "error: function 'broken' can exit without returning a value\n");
1078 DSLWriter::ProgramElements().clear();
1079 Var x(kFloat, "x");
1080 DSLFunction(kFloat, "broken").define(
1081 Declare(x, 0),
1082 If(x == 1, Return(x))
1083 );
1084 }
1085
1086 {
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001087 ExpectError error(r, "error: may not return a value from a void function\n");
1088 DSLWriter::ProgramElements().clear();
1089 DSLFunction(kVoid, "broken").define(
1090 Return(0)
1091 );
1092 }
1093
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001094 {
John Stilesb3dcbb12021-03-04 16:00:20 -05001095 ExpectError error(r, "error: function 'broken' can exit without returning a value\n");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001096 DSLWriter::ProgramElements().clear();
1097 DSLFunction(kFloat, "broken").define(
1098 );
1099 }
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001100}
1101
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001102DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLIf, r, ctxInfo) {
1103 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1104 Var a(kFloat, "a"), b(kFloat, "b");
1105 Statement x = If(a > b, a -= b);
John Stilesb4d7b582021-02-19 09:56:31 -05001106 EXPECT_EQUAL(x, "if ((a > b)) (a -= b);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001107
1108 Statement y = If(a > b, a -= b, b -= a);
John Stilesb4d7b582021-02-19 09:56:31 -05001109 EXPECT_EQUAL(y, "if ((a > b)) (a -= b); else (b -= a);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001110
1111 {
1112 ExpectError error(r, "error: expected 'bool', but found 'float'\n");
1113 If(a + b, a -= b).release();
1114 }
1115}
1116
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001117DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLReturn, r, ctxInfo) {
1118 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1119
1120 Statement x = Return();
John Stilesb4d7b582021-02-19 09:56:31 -05001121 EXPECT_EQUAL(x, "return;");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001122
1123 Statement y = Return(true);
John Stilesb4d7b582021-02-19 09:56:31 -05001124 EXPECT_EQUAL(y, "return true;");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001125}
1126
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001127DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSelect, r, ctxInfo) {
1128 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1129 Var a(kInt, "a");
1130 Expression x = Select(a > 0, 1, -1);
John Stilesb4d7b582021-02-19 09:56:31 -05001131 EXPECT_EQUAL(x, "((a > 0) ? 1 : -1)");
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001132
1133 {
1134 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001135 DSLExpression x = Select(a, 1, -1);
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001136 }
1137
1138 {
1139 ExpectError error(r, "error: ternary operator result mismatch: 'float2', 'float3'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001140 DSLExpression x = Select(a > 0, Float2(1), Float3(1));
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001141 }
1142}
1143
Ethan Nicholascfefec02021-02-09 15:22:57 -05001144DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSwitch, r, ctxInfo) {
1145 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1146
1147 Var a(kFloat, "a"), b(kInt, "b");
1148
1149 Statement x = Switch(5,
1150 Case(0, a = 0, Break()),
1151 Case(1, a = 1, Continue()),
John Stilese1d1b082021-02-23 13:44:36 -05001152 Case(2, a = 2 /*Fallthrough*/),
Ethan Nicholascfefec02021-02-09 15:22:57 -05001153 Default(Discard())
1154 );
John Stilese1d1b082021-02-23 13:44:36 -05001155 EXPECT_EQUAL(x, R"(
1156 switch (5) {
1157 case 0: (a = 0.0); break;
1158 case 1: (a = 1.0); continue;
1159 case 2: (a = 2.0);
1160 default: discard;
1161 }
1162 )");
Ethan Nicholascfefec02021-02-09 15:22:57 -05001163
John Stiles642cde22021-02-23 14:57:01 -05001164 EXPECT_EQUAL(Switch(b),
1165 "switch (b) {}");
1166
1167 EXPECT_EQUAL(Switch(b, Default(), Case(0), Case(1)),
1168 "switch (b) { default: case 0: case 1: }");
Ethan Nicholascfefec02021-02-09 15:22:57 -05001169
1170 {
John Stilese1d1b082021-02-23 13:44:36 -05001171 ExpectError error(r, "error: duplicate case value '0'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001172 DSLStatement(Switch(0, Case(0), Case(0))).release();
Ethan Nicholascfefec02021-02-09 15:22:57 -05001173 }
1174
1175 {
John Stilese1d1b082021-02-23 13:44:36 -05001176 ExpectError error(r, "error: duplicate default case\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001177 DSLStatement(Switch(0, Default(a = 0), Default(a = 1))).release();
John Stilese1d1b082021-02-23 13:44:36 -05001178 }
1179
1180 {
Ethan Nicholascfefec02021-02-09 15:22:57 -05001181 ExpectError error(r, "error: case value must be a constant integer\n");
1182 Var b(kInt);
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001183 DSLStatement(Switch(0, Case(b))).release();
Ethan Nicholascfefec02021-02-09 15:22:57 -05001184 }
1185
1186 {
1187 ExpectError error(r, "error: continue statement must be inside a loop\n");
1188 DSLFunction(kVoid, "fail").define(
1189 Switch(5,
1190 Case(0, a = 0, Break()),
1191 Case(1, a = 1, Continue()),
1192 Default(Discard())
1193 )
1194 );
1195 }
1196}
1197
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001198DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSwizzle, r, ctxInfo) {
1199 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1200 Var a(kFloat4, "a");
1201
1202 Expression e1 = a.x();
John Stilesb4d7b582021-02-19 09:56:31 -05001203 EXPECT_EQUAL(e1, "a.x");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001204
1205 Expression e2 = a.y();
John Stilesb4d7b582021-02-19 09:56:31 -05001206 EXPECT_EQUAL(e2, "a.y");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001207
1208 Expression e3 = a.z();
John Stilesb4d7b582021-02-19 09:56:31 -05001209 EXPECT_EQUAL(e3, "a.z");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001210
1211 Expression e4 = a.w();
John Stilesb4d7b582021-02-19 09:56:31 -05001212 EXPECT_EQUAL(e4, "a.w");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001213
1214 Expression e5 = a.r();
John Stilesb4d7b582021-02-19 09:56:31 -05001215 EXPECT_EQUAL(e5, "a.x");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001216
1217 Expression e6 = a.g();
John Stilesb4d7b582021-02-19 09:56:31 -05001218 EXPECT_EQUAL(e6, "a.y");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001219
1220 Expression e7 = a.b();
John Stilesb4d7b582021-02-19 09:56:31 -05001221 EXPECT_EQUAL(e7, "a.z");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001222
1223 Expression e8 = a.a();
John Stilesb4d7b582021-02-19 09:56:31 -05001224 EXPECT_EQUAL(e8, "a.w");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001225
1226 Expression e9 = Swizzle(a, R);
John Stilesb4d7b582021-02-19 09:56:31 -05001227 EXPECT_EQUAL(e9, "a.x");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001228
1229 Expression e10 = Swizzle(a, ZERO, G);
John Stiles54f00492021-02-19 11:46:10 -05001230 EXPECT_EQUAL(e10, "float2(a.y, 0.0).yx");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001231
1232 Expression e11 = Swizzle(a, B, G, G);
John Stilesb4d7b582021-02-19 09:56:31 -05001233 EXPECT_EQUAL(e11, "a.zyy");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001234
1235 Expression e12 = Swizzle(a, R, G, B, ONE);
John Stiles54f00492021-02-19 11:46:10 -05001236 EXPECT_EQUAL(e12, "float4(a.xyz, 1.0)");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001237
1238 Expression e13 = Swizzle(a, R, G, B, ONE).r();
John Stiles54f00492021-02-19 11:46:10 -05001239 EXPECT_EQUAL(e13, "float4(a.xyz, 1.0).x");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001240}
1241
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001242DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLWhile, r, ctxInfo) {
1243 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1244 Statement x = While(true, Block());
John Stilesb4d7b582021-02-19 09:56:31 -05001245 EXPECT_EQUAL(x, "for (; true;) {}");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001246
1247 Var a(kFloat, "a"), b(kFloat, "b");
1248 Statement y = While(a != b, Block(a++, --b));
John Stilesb4d7b582021-02-19 09:56:31 -05001249 EXPECT_EQUAL(y, "for (; (a != b);) { a++; --b; }");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001250
1251 {
1252 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001253 DSLStatement x = While(7, Block());
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001254 }
1255}
Ethan Nicholas04be3392021-01-26 10:07:01 -05001256
1257DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLIndex, r, ctxInfo) {
1258 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1259 Var a(Array(kInt, 5), "a"), b(kInt, "b");
John Stilesb4d7b582021-02-19 09:56:31 -05001260
1261 EXPECT_EQUAL(a[0], "a[0]");
1262 EXPECT_EQUAL(a[b], "a[b]");
Ethan Nicholas04be3392021-01-26 10:07:01 -05001263
1264 {
1265 ExpectError error(r, "error: expected 'int', but found 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001266 DSLExpression x = a[true];
Ethan Nicholas04be3392021-01-26 10:07:01 -05001267 }
1268
1269 {
1270 ExpectError error(r, "error: expected array, but found 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001271 DSLExpression x = b[0];
Ethan Nicholas04be3392021-01-26 10:07:01 -05001272 }
1273
1274 {
1275 ExpectError error(r, "error: index -1 out of range for 'int[5]'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001276 DSLExpression x = a[-1];
Ethan Nicholas04be3392021-01-26 10:07:01 -05001277 }
1278}
Ethan Nicholas30e93d52021-01-26 12:00:25 -05001279
1280DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBuiltins, r, ctxInfo) {
1281 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1282 // There is a Fract type on Mac which can conflict with our Fract builtin
1283 using SkSL::dsl::Fract;
1284 Var a(kHalf4, "a"), b(kHalf4, "b"), c(kHalf4, "c");
1285 Var h3(kHalf3, "h3");
1286 Var b4(kBool4, "b4");
John Stilesb4d7b582021-02-19 09:56:31 -05001287 EXPECT_EQUAL(Abs(a), "abs(a)");
1288 EXPECT_EQUAL(All(b4), "all(b4)");
1289 EXPECT_EQUAL(Any(b4), "any(b4)");
1290 EXPECT_EQUAL(Ceil(a), "ceil(a)");
1291 EXPECT_EQUAL(Clamp(a, 0, 1), "clamp(a, 0.0, 1.0)");
1292 EXPECT_EQUAL(Cos(a), "cos(a)");
1293 EXPECT_EQUAL(Cross(h3, h3), "cross(h3, h3)");
1294 EXPECT_EQUAL(Degrees(a), "degrees(a)");
1295 EXPECT_EQUAL(Distance(a, b), "distance(a, b)");
1296 EXPECT_EQUAL(Dot(a, b), "dot(a, b)");
1297 EXPECT_EQUAL(Equal(a, b), "equal(a, b)");
1298 EXPECT_EQUAL(Exp(a), "exp(a)");
1299 EXPECT_EQUAL(Exp2(a), "exp2(a)");
1300 EXPECT_EQUAL(Faceforward(a, b, c), "faceforward(a, b, c)");
1301 EXPECT_EQUAL(Floor(a), "floor(a)");
1302 EXPECT_EQUAL(Fract(a), "fract(a)");
1303 EXPECT_EQUAL(GreaterThan(a, b), "greaterThan(a, b)");
1304 EXPECT_EQUAL(GreaterThanEqual(a, b), "greaterThanEqual(a, b)");
1305 EXPECT_EQUAL(Inversesqrt(a), "inversesqrt(a)");
1306 EXPECT_EQUAL(LessThan(a, b), "lessThan(a, b)");
1307 EXPECT_EQUAL(LessThanEqual(a, b), "lessThanEqual(a, b)");
1308 EXPECT_EQUAL(Length(a), "length(a)");
1309 EXPECT_EQUAL(Log(a), "log(a)");
1310 EXPECT_EQUAL(Log2(a), "log2(a)");
1311 EXPECT_EQUAL(Max(a, b), "max(a, b)");
1312 EXPECT_EQUAL(Min(a, b), "min(a, b)");
1313 EXPECT_EQUAL(Mix(a, b, c), "mix(a, b, c)");
1314 EXPECT_EQUAL(Mod(a, b), "mod(a, b)");
1315 EXPECT_EQUAL(Normalize(a), "normalize(a)");
1316 EXPECT_EQUAL(NotEqual(a, b), "notEqual(a, b)");
1317 EXPECT_EQUAL(Pow(a, b), "pow(a, b)");
1318 EXPECT_EQUAL(Radians(a), "radians(a)");
1319 EXPECT_EQUAL(Reflect(a, b), "reflect(a, b)");
1320 EXPECT_EQUAL(Refract(a, b, 1), "refract(a, b, 1.0)");
1321 EXPECT_EQUAL(Saturate(a), "saturate(a)");
1322 EXPECT_EQUAL(Sign(a), "sign(a)");
1323 EXPECT_EQUAL(Sin(a), "sin(a)");
1324 EXPECT_EQUAL(Smoothstep(a, b, c), "smoothstep(a, b, c)");
1325 EXPECT_EQUAL(Sqrt(a), "sqrt(a)");
1326 EXPECT_EQUAL(Step(a, b), "step(a, b)");
1327 EXPECT_EQUAL(Tan(a), "tan(a)");
1328 EXPECT_EQUAL(Unpremul(a), "unpremul(a)");
Ethan Nicholas30e93d52021-01-26 12:00:25 -05001329
1330 // these calls all go through the normal channels, so it ought to be sufficient to prove that
1331 // one of them reports errors correctly
1332 {
1333 ExpectError error(r, "error: no match for ceil(bool)\n");
1334 Ceil(a == b).release();
1335 }
1336}
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001337
1338DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLModifiers, r, ctxInfo) {
1339 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1340
1341 Var v1(kConst_Modifier, kInt, "v1");
Ethan Nicholasbd974002021-02-22 16:20:06 -05001342 Statement d1 = Declare(v1, 0);
1343 EXPECT_EQUAL(d1, "const int v1 = 0;");
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001344
1345 // Most modifiers require an appropriate context to be legal. We can't yet give them that
1346 // context, so we can't as yet Declare() variables with these modifiers.
1347 // TODO: better tests when able
1348 Var v2(kIn_Modifier, kInt, "v2");
1349 REPORTER_ASSERT(r, DSLWriter::Var(v2).modifiers().fFlags == SkSL::Modifiers::kIn_Flag);
1350
1351 Var v3(kOut_Modifier, kInt, "v3");
1352 REPORTER_ASSERT(r, DSLWriter::Var(v3).modifiers().fFlags == SkSL::Modifiers::kOut_Flag);
1353
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001354 Var v4(kFlat_Modifier, kInt, "v4");
1355 REPORTER_ASSERT(r, DSLWriter::Var(v4).modifiers().fFlags == SkSL::Modifiers::kFlat_Flag);
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001356
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001357 Var v5(kNoPerspective_Modifier, kInt, "v5");
1358 REPORTER_ASSERT(r, DSLWriter::Var(v5).modifiers().fFlags ==
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001359 SkSL::Modifiers::kNoPerspective_Flag);
1360
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001361 Var v6(kIn_Modifier | kOut_Modifier, kInt, "v6");
1362 REPORTER_ASSERT(r, DSLWriter::Var(v6).modifiers().fFlags ==
1363 (SkSL::Modifiers::kIn_Flag | SkSL::Modifiers::kOut_Flag));
1364
1365 Var v7(kInOut_Modifier, kInt, "v7");
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001366 REPORTER_ASSERT(r, DSLWriter::Var(v7).modifiers().fFlags ==
1367 (SkSL::Modifiers::kIn_Flag | SkSL::Modifiers::kOut_Flag));
1368
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001369 Var v8(kUniform_Modifier, kInt, "v8");
1370 REPORTER_ASSERT(r, DSLWriter::Var(v8).modifiers().fFlags == SkSL::Modifiers::kUniform_Flag);
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001371}
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001372
1373DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLStruct, r, ctxInfo) {
1374 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1375
1376 DSLType simpleStruct = Struct("SimpleStruct",
1377 Field(kFloat, "x"),
1378 Field(kBool, "b"),
1379 Field(Array(kFloat, 3), "a")
1380 );
1381 DSLVar result(simpleStruct, "result");
1382 DSLFunction(simpleStruct, "returnStruct").define(
1383 Declare(result),
1384 result.field("x") = 123,
1385 result.field("b") = result.field("x") > 0,
1386 result.field("a")[0] = result.field("x"),
1387 Return(result)
1388 );
1389 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 2);
John Stilesb4d7b582021-02-19 09:56:31 -05001390 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1391 "struct SimpleStruct { float x; bool b; float[3] a; };");
1392 EXPECT_EQUAL(*DSLWriter::ProgramElements()[1],
1393 "SimpleStruct returnStruct() { SimpleStruct result; (result.x = 123.0);"
1394 "(result.b = (result.x > 0.0)); (result.a[0] = result.x); return result; }");
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001395
1396 DSLWriter::ProgramElements().clear();
1397 Struct("NestedStruct",
1398 Field(kInt, "x"),
1399 Field(simpleStruct, "simple")
1400 );
1401 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
John Stilesb4d7b582021-02-19 09:56:31 -05001402 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1403 "struct NestedStruct { int x; SimpleStruct simple; };");
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001404}