blob: 2201da6767a594cd4126ff530f51df18f472fa9e [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 Nicholas24c17722021-03-09 13:10:59 -05008#include "include/private/SkSLIRNode.h"
Ethan Nicholasdaed2592021-03-04 14:30:25 -05009#include "include/sksl/DSL.h"
Ethan Nicholas95046142021-01-07 10:57:27 -050010#include "src/gpu/GrDirectContextPriv.h"
11#include "src/gpu/GrGpu.h"
12#include "src/sksl/SkSLIRGenerator.h"
Ethan Nicholas95046142021-01-07 10:57:27 -050013#include "src/sksl/dsl/priv/DSLWriter.h"
14
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
Ethan Nicholas55a63af2021-05-18 10:12:58 -040021SkSL::ProgramSettings default_settings() {
22 SkSL::ProgramSettings result;
23 result.fDSLMarkVarsDeclared = true;
24 result.fDSLMangling = false;
25 return result;
26}
27
28SkSL::ProgramSettings no_mark_vars_declared() {
29 SkSL::ProgramSettings result = default_settings();
30 result.fDSLMarkVarsDeclared = false;
31 return result;
32}
Ethan Nicholas22dcb732021-05-12 11:26:58 -040033
Ethan Nicholas961d9442021-03-16 16:37:29 -040034/**
35 * In addition to issuing an automatic Start() and End(), disables mangling and optionally
36 * auto-declares variables during its lifetime. Variable auto-declaration simplifies testing so we
37 * don't have to sprinkle all the tests with a bunch of Declare(foo).release() calls just to avoid
38 * errors, especially given that some of the variables have options that make them an error to
39 * actually declare.
40 */
Ethan Nicholas95046142021-01-07 10:57:27 -050041class AutoDSLContext {
42public:
Ethan Nicholas55a63af2021-05-18 10:12:58 -040043 AutoDSLContext(GrGpu* gpu, SkSL::ProgramSettings settings = default_settings(),
Ethan Nicholasee49efc2021-04-09 15:33:53 -040044 SkSL::ProgramKind kind = SkSL::ProgramKind::kFragment) {
Ethan Nicholas55a63af2021-05-18 10:12:58 -040045 Start(gpu->shaderCompiler(), kind, settings);
Ethan Nicholas95046142021-01-07 10:57:27 -050046 }
47
48 ~AutoDSLContext() {
49 End();
50 }
51};
52
Ethan Nicholasb3d4e742021-01-08 11:42:25 -050053class ExpectError : public ErrorHandler {
54public:
55 ExpectError(skiatest::Reporter* reporter, const char* msg)
56 : fMsg(msg)
57 , fReporter(reporter) {
58 SetErrorHandler(this);
59 }
60
61 ~ExpectError() override {
John Stiles642cde22021-02-23 14:57:01 -050062 REPORTER_ASSERT(fReporter, !fMsg,
63 "Error mismatch: expected:\n%sbut no error occurred\n", fMsg);
Ethan Nicholasb3d4e742021-01-08 11:42:25 -050064 SetErrorHandler(nullptr);
65 }
66
Ethan Nicholasb9563042021-02-25 09:45:49 -050067 void handleError(const char* msg, PositionInfo* pos) override {
Ethan Nicholasb3d4e742021-01-08 11:42:25 -050068 REPORTER_ASSERT(fReporter, !strcmp(msg, fMsg),
69 "Error mismatch: expected:\n%sbut received:\n%s", fMsg, msg);
70 fMsg = nullptr;
71 }
72
73private:
74 const char* fMsg;
75 skiatest::Reporter* fReporter;
76};
77
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -050078static bool whitespace_insensitive_compare(const char* a, const char* b) {
79 for (;;) {
80 while (isspace(*a)) {
81 ++a;
82 }
83 while (isspace(*b)) {
84 ++b;
85 }
86 if (*a != *b) {
87 return false;
88 }
89 if (*a == 0) {
90 return true;
91 }
92 ++a;
93 ++b;
94 }
95}
96
Ethan Nicholas24c17722021-03-09 13:10:59 -050097// for use from SkSLDSLOnlyTest.cpp
98void StartDSL(const sk_gpu_test::ContextInfo ctxInfo) {
99 Start(ctxInfo.directContext()->priv().getGpu()->shaderCompiler());
100}
101
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500102DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLStartup, r, ctxInfo) {
Ethan Nicholas95046142021-01-07 10:57:27 -0500103 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
104 Expression e1 = 1;
105 REPORTER_ASSERT(r, e1.release()->description() == "1");
106 Expression e2 = 1.0;
107 REPORTER_ASSERT(r, e2.release()->description() == "1.0");
108 Expression e3 = true;
109 REPORTER_ASSERT(r, e3.release()->description() == "true");
Ethan Nicholasb14e6b92021-04-08 16:56:05 -0400110 Var a(kInt_Type, "a");
Ethan Nicholasbffe80a2021-01-11 15:42:44 -0500111 Expression e4 = a;
112 REPORTER_ASSERT(r, e4.release()->description() == "a");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -0500113
114 REPORTER_ASSERT(r, whitespace_insensitive_compare("", ""));
115 REPORTER_ASSERT(r, !whitespace_insensitive_compare("", "a"));
116 REPORTER_ASSERT(r, !whitespace_insensitive_compare("a", ""));
117 REPORTER_ASSERT(r, whitespace_insensitive_compare("a", "a"));
118 REPORTER_ASSERT(r, whitespace_insensitive_compare("abc", "abc"));
119 REPORTER_ASSERT(r, whitespace_insensitive_compare("abc", " abc "));
120 REPORTER_ASSERT(r, whitespace_insensitive_compare("a b c ", "\n\n\nabc"));
121 REPORTER_ASSERT(r, !whitespace_insensitive_compare("a b c d", "\n\n\nabc"));
Ethan Nicholas95046142021-01-07 10:57:27 -0500122}
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500123
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500124static SkSL::String stringize(DSLStatement& stmt) { return stmt.release()->description(); }
125static SkSL::String stringize(DSLPossibleStatement& stmt) { return stmt.release()->description(); }
126static SkSL::String stringize(DSLExpression& expr) { return expr.release()->description(); }
127static SkSL::String stringize(DSLPossibleExpression& expr) { return expr.release()->description(); }
Ethan Nicholas722cb672021-05-06 10:47:06 -0400128static SkSL::String stringize(DSLBlock& blck) { return blck.release()->description(); }
Ethan Nicholas292a09d2021-07-14 09:52:16 -0400129static SkSL::String stringize(SkSL::IRNode& node) { return node.description(); }
130static SkSL::String stringize(SkSL::Program& program) { return program.description(); }
John Stilesb4d7b582021-02-19 09:56:31 -0500131
132template <typename T>
133static void expect_equal(skiatest::Reporter* r, int lineNumber, T& input, const char* expected) {
134 SkSL::String actual = stringize(input);
135 if (!whitespace_insensitive_compare(expected, actual.c_str())) {
136 ERRORF(r, "(Failed on line %d)\nExpected: %s\n Actual: %s\n",
137 lineNumber, expected, actual.c_str());
138 }
139}
140
141template <typename T>
142static void expect_equal(skiatest::Reporter* r, int lineNumber, T&& dsl, const char* expected) {
143 // This overload allows temporary values to be passed to expect_equal.
144 return expect_equal(r, lineNumber, dsl, expected);
145}
146
147#define EXPECT_EQUAL(a, b) expect_equal(r, __LINE__, (a), (b))
148
Ethan Nicholas22dcb732021-05-12 11:26:58 -0400149DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLFlags, r, ctxInfo) {
150 {
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400151 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), no_mark_vars_declared());
Ethan Nicholas22dcb732021-05-12 11:26:58 -0400152 EXPECT_EQUAL(All(GreaterThan(Float4(1), Float4(0))), "true");
153
154 Var x(kInt_Type, "x");
155 EXPECT_EQUAL(Declare(x), "int x;");
156 }
157
158 {
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400159 SkSL::ProgramSettings settings;
160 settings.fOptimize = false;
161 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), settings,
Brian Osman8c264792021-07-01 16:41:27 -0400162 SkSL::ProgramKind::kFragment);
Ethan Nicholas22dcb732021-05-12 11:26:58 -0400163 EXPECT_EQUAL(All(GreaterThan(Float4(1), Float4(0))),
164 "all(greaterThan(float4(1.0), float4(0.0)))");
165 }
166
167 {
Ethan Nicholas55a63af2021-05-18 10:12:58 -0400168 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), SkSL::ProgramSettings());
Ethan Nicholas22dcb732021-05-12 11:26:58 -0400169 Var x(kInt_Type, "x");
170 EXPECT_EQUAL(Declare(x), "int _0_x;");
171 }
172}
173
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500174DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLFloat, r, ctxInfo) {
175 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
176 Expression e1 = Float(std::numeric_limits<float>::max());
177 REPORTER_ASSERT(r, atof(e1.release()->description().c_str()) ==
178 std::numeric_limits<float>::max());
179
180 Expression e2 = Float(std::numeric_limits<float>::min());
181 REPORTER_ASSERT(r, atof(e2.release()->description().c_str()) ==
182 std::numeric_limits<float>::min());
183
John Stilesb4d7b582021-02-19 09:56:31 -0500184 EXPECT_EQUAL(Float2(0),
185 "float2(0.0)");
186 EXPECT_EQUAL(Float2(-0.5, 1),
187 "float2(-0.5, 1.0)");
188 EXPECT_EQUAL(Float3(0.75),
189 "float3(0.75)");
190 EXPECT_EQUAL(Float3(Float2(0, 1), -2),
John Stilesb9e4f642021-03-05 09:11:38 -0500191 "float3(0.0, 1.0, -2.0)");
John Stilesb4d7b582021-02-19 09:56:31 -0500192 EXPECT_EQUAL(Float3(0, 1, 2),
193 "float3(0.0, 1.0, 2.0)");
194 EXPECT_EQUAL(Float4(0),
195 "float4(0.0)");
196 EXPECT_EQUAL(Float4(Float2(0, 1), Float2(2, 3)),
John Stilesb9e4f642021-03-05 09:11:38 -0500197 "float4(0.0, 1.0, 2.0, 3.0)");
John Stilesb4d7b582021-02-19 09:56:31 -0500198 EXPECT_EQUAL(Float4(0, 1, Float2(2, 3)),
John Stilesb9e4f642021-03-05 09:11:38 -0500199 "float4(0.0, 1.0, 2.0, 3.0)");
John Stilesb4d7b582021-02-19 09:56:31 -0500200 EXPECT_EQUAL(Float4(0, 1, 2, 3),
201 "float4(0.0, 1.0, 2.0, 3.0)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500202
Ethan Nicholasa60cc3e2021-04-23 16:15:11 -0400203 DSLVar x(kFloat_Type, "x");
204 EXPECT_EQUAL(x = 1.0, "(x = 1.0)");
205 EXPECT_EQUAL(x = 1.0f, "(x = 1.0)");
206
207 DSLVar y(kFloat2_Type, "y");
208 EXPECT_EQUAL(y.x() = 1.0, "(y.x = 1.0)");
209 EXPECT_EQUAL(y.x() = 1.0f, "(y.x = 1.0)");
210
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500211 {
212 ExpectError error(r, "error: floating point value is infinite\n");
213 Float(std::numeric_limits<float>::infinity()).release();
214 }
215
216 {
217 ExpectError error(r, "error: floating point value is NaN\n");
218 Float(std::numeric_limits<float>::quiet_NaN()).release();
219 }
220
221 {
John Stiles7729e0a2021-07-13 09:18:57 -0400222 ExpectError error(r, "error: 'float4' is not a valid parameter to 'float2' constructor; "
223 "use '.xy' instead\n");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500224 Float2(Float4(1)).release();
225 }
226
227 {
228 ExpectError error(r, "error: invalid arguments to 'float4' constructor (expected 4 scalars,"
229 " but found 3)\n");
230 Float4(Float3(1)).release();
231 }
232}
233
234DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLHalf, r, ctxInfo) {
235 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
236 Expression e1 = Half(std::numeric_limits<float>::max());
John Stilesb4d7b582021-02-19 09:56:31 -0500237 REPORTER_ASSERT(r,
238 atof(e1.release()->description().c_str()) == std::numeric_limits<float>::max());
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500239
240 Expression e2 = Half(std::numeric_limits<float>::min());
John Stilesb4d7b582021-02-19 09:56:31 -0500241 REPORTER_ASSERT(r,
242 atof(e2.release()->description().c_str()) == std::numeric_limits<float>::min());
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500243
John Stilesb9e4f642021-03-05 09:11:38 -0500244 EXPECT_EQUAL(Half2(0),
245 "half2(0.0)");
246 EXPECT_EQUAL(Half2(-0.5, 1),
247 "half2(-0.5, 1.0)");
248 EXPECT_EQUAL(Half3(0.75),
249 "half3(0.75)");
250 EXPECT_EQUAL(Half3(Half2(0, 1), -2),
251 "half3(0.0, 1.0, -2.0)");
252 EXPECT_EQUAL(Half3(0, 1, 2),
253 "half3(0.0, 1.0, 2.0)");
254 EXPECT_EQUAL(Half4(0),
255 "half4(0.0)");
256 EXPECT_EQUAL(Half4(Half2(0, 1), Half2(2, 3)),
257 "half4(0.0, 1.0, 2.0, 3.0)");
258 EXPECT_EQUAL(Half4(0, 1, Half2(2, 3)),
259 "half4(0.0, 1.0, 2.0, 3.0)");
260 EXPECT_EQUAL(Half4(0, 1, 2, 3),
261 "half4(0.0, 1.0, 2.0, 3.0)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500262
263 {
264 ExpectError error(r, "error: floating point value is infinite\n");
265 Half(std::numeric_limits<float>::infinity()).release();
266 }
267
268 {
269 ExpectError error(r, "error: floating point value is NaN\n");
270 Half(std::numeric_limits<float>::quiet_NaN()).release();
271 }
272
273 {
John Stiles7729e0a2021-07-13 09:18:57 -0400274 ExpectError error(r, "error: 'half4' is not a valid parameter to 'half2' constructor; use "
275 "'.xy' instead\n");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500276 Half2(Half4(1)).release();
277 }
278
279 {
280 ExpectError error(r, "error: invalid arguments to 'half4' constructor (expected 4 scalars,"
281 " but found 3)\n");
282 Half4(Half3(1)).release();
283 }
284}
285
286DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLInt, r, ctxInfo) {
287 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500288
John Stilesb9e4f642021-03-05 09:11:38 -0500289 EXPECT_EQUAL(Int(std::numeric_limits<int32_t>::max()),
290 "2147483647");
291 EXPECT_EQUAL(Int2(std::numeric_limits<int32_t>::min()),
292 "int2(-2147483648)");
293 EXPECT_EQUAL(Int2(0, 1),
294 "int2(0, 1)");
295 EXPECT_EQUAL(Int3(0),
296 "int3(0)");
297 EXPECT_EQUAL(Int3(Int2(0, 1), -2),
298 "int3(0, 1, -2)");
299 EXPECT_EQUAL(Int3(0, 1, 2),
300 "int3(0, 1, 2)");
301 EXPECT_EQUAL(Int4(0),
302 "int4(0)");
303 EXPECT_EQUAL(Int4(Int2(0, 1), Int2(2, 3)),
304 "int4(0, 1, 2, 3)");
305 EXPECT_EQUAL(Int4(0, 1, Int2(2, 3)),
306 "int4(0, 1, 2, 3)");
307 EXPECT_EQUAL(Int4(0, 1, 2, 3),
308 "int4(0, 1, 2, 3)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500309
310 {
John Stiles7729e0a2021-07-13 09:18:57 -0400311 ExpectError error(r, "error: 'int4' is not a valid parameter to 'int2' constructor; use "
312 "'.xy' instead\n");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500313 Int2(Int4(1)).release();
314 }
315
316 {
317 ExpectError error(r, "error: invalid arguments to 'int4' constructor (expected 4 scalars,"
318 " but found 3)\n");
319 Int4(Int3(1)).release();
320 }
321}
322
Ethan Nicholasb83199e2021-05-03 14:25:35 -0400323DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLUInt, r, ctxInfo) {
324 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
325
326 EXPECT_EQUAL(UInt(std::numeric_limits<uint32_t>::max()),
327 "4294967295");
328 EXPECT_EQUAL(UInt2(std::numeric_limits<uint32_t>::min()),
329 "uint2(0)");
330 EXPECT_EQUAL(UInt2(0, 1),
331 "uint2(0, 1)");
332 EXPECT_EQUAL(UInt3(0),
333 "uint3(0)");
334 EXPECT_EQUAL(UInt3(UInt2(0, 1), -2),
335 "uint3(0, 1, -2)");
336 EXPECT_EQUAL(UInt3(0, 1, 2),
337 "uint3(0, 1, 2)");
338 EXPECT_EQUAL(UInt4(0),
339 "uint4(0)");
340 EXPECT_EQUAL(UInt4(UInt2(0, 1), UInt2(2, 3)),
341 "uint4(0, 1, 2, 3)");
342 EXPECT_EQUAL(UInt4(0, 1, UInt2(2, 3)),
343 "uint4(0, 1, 2, 3)");
344 EXPECT_EQUAL(UInt4(0, 1, 2, 3),
345 "uint4(0, 1, 2, 3)");
346
347 {
John Stiles7729e0a2021-07-13 09:18:57 -0400348 ExpectError error(r, "error: 'uint4' is not a valid parameter to 'uint2' constructor; use "
349 "'.xy' instead\n");
Ethan Nicholasb83199e2021-05-03 14:25:35 -0400350 UInt2(UInt4(1)).release();
351 }
352
353 {
354 ExpectError error(r, "error: invalid arguments to 'uint4' constructor (expected 4 scalars,"
355 " but found 3)\n");
356 UInt4(UInt3(1)).release();
357 }
358}
359
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500360DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLShort, r, ctxInfo) {
361 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500362
John Stilesb9e4f642021-03-05 09:11:38 -0500363 EXPECT_EQUAL(Short(std::numeric_limits<int16_t>::max()),
364 "32767");
365 EXPECT_EQUAL(Short2(std::numeric_limits<int16_t>::min()),
366 "short2(-32768)");
367 EXPECT_EQUAL(Short2(0, 1),
368 "short2(0, 1)");
369 EXPECT_EQUAL(Short3(0),
370 "short3(0)");
371 EXPECT_EQUAL(Short3(Short2(0, 1), -2),
372 "short3(0, 1, -2)");
373 EXPECT_EQUAL(Short3(0, 1, 2),
374 "short3(0, 1, 2)");
375 EXPECT_EQUAL(Short4(0),
376 "short4(0)");
377 EXPECT_EQUAL(Short4(Short2(0, 1), Short2(2, 3)),
378 "short4(0, 1, 2, 3)");
379 EXPECT_EQUAL(Short4(0, 1, Short2(2, 3)),
380 "short4(0, 1, 2, 3)");
381 EXPECT_EQUAL(Short4(0, 1, 2, 3),
382 "short4(0, 1, 2, 3)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500383
384 {
John Stiles7729e0a2021-07-13 09:18:57 -0400385 ExpectError error(r, "error: 'short4' is not a valid parameter to 'short2' constructor; "
386 "use '.xy' instead\n");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500387 Short2(Short4(1)).release();
388 }
389
390 {
391 ExpectError error(r, "error: invalid arguments to 'short4' constructor (expected 4 scalars,"
392 " but found 3)\n");
393 Short4(Short3(1)).release();
394 }
395}
396
Ethan Nicholasb83199e2021-05-03 14:25:35 -0400397DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLUShort, r, ctxInfo) {
398 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
399
400 EXPECT_EQUAL(UShort(std::numeric_limits<uint16_t>::max()),
401 "65535");
402 EXPECT_EQUAL(UShort2(std::numeric_limits<uint16_t>::min()),
403 "ushort2(0)");
404 EXPECT_EQUAL(UShort2(0, 1),
405 "ushort2(0, 1)");
406 EXPECT_EQUAL(UShort3(0),
407 "ushort3(0)");
408 EXPECT_EQUAL(UShort3(UShort2(0, 1), -2),
409 "ushort3(0, 1, -2)");
410 EXPECT_EQUAL(UShort3(0, 1, 2),
411 "ushort3(0, 1, 2)");
412 EXPECT_EQUAL(UShort4(0),
413 "ushort4(0)");
414 EXPECT_EQUAL(UShort4(UShort2(0, 1), UShort2(2, 3)),
415 "ushort4(0, 1, 2, 3)");
416 EXPECT_EQUAL(UShort4(0, 1, UShort2(2, 3)),
417 "ushort4(0, 1, 2, 3)");
418 EXPECT_EQUAL(UShort4(0, 1, 2, 3),
419 "ushort4(0, 1, 2, 3)");
420
421 {
John Stiles7729e0a2021-07-13 09:18:57 -0400422 ExpectError error(r, "error: 'ushort4' is not a valid parameter to 'ushort2' constructor; "
423 "use '.xy' instead\n");
Ethan Nicholasb83199e2021-05-03 14:25:35 -0400424 UShort2(UShort4(1)).release();
425 }
426
427 {
428 ExpectError error(r, "error: invalid arguments to 'ushort4' constructor (expected 4 "
429 "scalars, but found 3)\n");
430 UShort4(UShort3(1)).release();
431 }
432}
433
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500434DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBool, r, ctxInfo) {
435 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500436
John Stilesb9e4f642021-03-05 09:11:38 -0500437 EXPECT_EQUAL(Bool2(false),
438 "bool2(false)");
439 EXPECT_EQUAL(Bool2(false, true),
440 "bool2(false, true)");
441 EXPECT_EQUAL(Bool3(false),
442 "bool3(false)");
443 EXPECT_EQUAL(Bool3(Bool2(false, true), false),
444 "bool3(false, true, false)");
445 EXPECT_EQUAL(Bool3(false, true, false),
446 "bool3(false, true, false)");
447 EXPECT_EQUAL(Bool4(false),
448 "bool4(false)");
449 EXPECT_EQUAL(Bool4(Bool2(false, true), Bool2(false, true)),
450 "bool4(false, true, false, true)");
451 EXPECT_EQUAL(Bool4(false, true, Bool2(false, true)),
452 "bool4(false, true, false, true)");
453 EXPECT_EQUAL(Bool4(false, true, false, true),
454 "bool4(false, true, false, true)");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500455
456 {
John Stiles7729e0a2021-07-13 09:18:57 -0400457 ExpectError error(r, "error: 'bool4' is not a valid parameter to 'bool2' constructor; use "
458 "'.xy' instead\n");
Ethan Nicholasb3d4e742021-01-08 11:42:25 -0500459 Bool2(Bool4(true)).release();
460 }
461
462 {
463 ExpectError error(r, "error: invalid arguments to 'bool4' constructor (expected 4 scalars,"
464 " but found 3)\n");
465 Bool4(Bool3(true)).release();
466 }
467}
Ethan Nicholas92969f22021-01-13 10:38:59 -0500468
Ethan Nicholasb83199e2021-05-03 14:25:35 -0400469DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLType, r, ctxInfo) {
470 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
471 REPORTER_ASSERT(r, DSLType(kBool_Type).isBoolean());
472 REPORTER_ASSERT(r, !DSLType(kBool_Type).isNumber());
473 REPORTER_ASSERT(r, !DSLType(kBool_Type).isFloat());
474 REPORTER_ASSERT(r, !DSLType(kBool_Type).isSigned());
475 REPORTER_ASSERT(r, !DSLType(kBool_Type).isUnsigned());
476 REPORTER_ASSERT(r, !DSLType(kBool_Type).isInteger());
477 REPORTER_ASSERT(r, DSLType(kBool_Type).isScalar());
478 REPORTER_ASSERT(r, !DSLType(kBool_Type).isVector());
479 REPORTER_ASSERT(r, !DSLType(kBool_Type).isMatrix());
480 REPORTER_ASSERT(r, !DSLType(kBool_Type).isArray());
481 REPORTER_ASSERT(r, !DSLType(kBool_Type).isStruct());
482
483 REPORTER_ASSERT(r, !DSLType(kInt_Type).isBoolean());
484 REPORTER_ASSERT(r, DSLType(kInt_Type).isNumber());
485 REPORTER_ASSERT(r, !DSLType(kInt_Type).isFloat());
486 REPORTER_ASSERT(r, DSLType(kInt_Type).isSigned());
487 REPORTER_ASSERT(r, !DSLType(kInt_Type).isUnsigned());
488 REPORTER_ASSERT(r, DSLType(kInt_Type).isInteger());
489 REPORTER_ASSERT(r, DSLType(kInt_Type).isScalar());
490 REPORTER_ASSERT(r, !DSLType(kInt_Type).isVector());
491 REPORTER_ASSERT(r, !DSLType(kInt_Type).isMatrix());
492 REPORTER_ASSERT(r, !DSLType(kInt_Type).isArray());
493 REPORTER_ASSERT(r, !DSLType(kInt_Type).isStruct());
494
495 REPORTER_ASSERT(r, !DSLType(kUInt_Type).isBoolean());
496 REPORTER_ASSERT(r, DSLType(kUInt_Type).isNumber());
497 REPORTER_ASSERT(r, !DSLType(kUInt_Type).isFloat());
498 REPORTER_ASSERT(r, !DSLType(kUInt_Type).isSigned());
499 REPORTER_ASSERT(r, DSLType(kUInt_Type).isUnsigned());
500 REPORTER_ASSERT(r, DSLType(kUInt_Type).isInteger());
501 REPORTER_ASSERT(r, DSLType(kUInt_Type).isScalar());
502 REPORTER_ASSERT(r, !DSLType(kUInt_Type).isVector());
503 REPORTER_ASSERT(r, !DSLType(kUInt_Type).isMatrix());
504 REPORTER_ASSERT(r, !DSLType(kUInt_Type).isArray());
505 REPORTER_ASSERT(r, !DSLType(kUInt_Type).isStruct());
506
507 REPORTER_ASSERT(r, !DSLType(kFloat_Type).isBoolean());
508 REPORTER_ASSERT(r, DSLType(kFloat_Type).isNumber());
509 REPORTER_ASSERT(r, DSLType(kFloat_Type).isFloat());
510 REPORTER_ASSERT(r, !DSLType(kFloat_Type).isSigned());
511 REPORTER_ASSERT(r, !DSLType(kFloat_Type).isUnsigned());
512 REPORTER_ASSERT(r, !DSLType(kFloat_Type).isInteger());
513 REPORTER_ASSERT(r, DSLType(kFloat_Type).isScalar());
514 REPORTER_ASSERT(r, !DSLType(kFloat_Type).isVector());
515 REPORTER_ASSERT(r, !DSLType(kFloat_Type).isMatrix());
516 REPORTER_ASSERT(r, !DSLType(kFloat_Type).isArray());
517 REPORTER_ASSERT(r, !DSLType(kFloat_Type).isStruct());
518
519 REPORTER_ASSERT(r, !DSLType(kFloat2_Type).isBoolean());
520 REPORTER_ASSERT(r, !DSLType(kFloat2_Type).isNumber());
521 REPORTER_ASSERT(r, !DSLType(kFloat2_Type).isFloat());
522 REPORTER_ASSERT(r, !DSLType(kFloat2_Type).isSigned());
523 REPORTER_ASSERT(r, !DSLType(kFloat2_Type).isUnsigned());
524 REPORTER_ASSERT(r, !DSLType(kFloat2_Type).isInteger());
525 REPORTER_ASSERT(r, !DSLType(kFloat2_Type).isScalar());
526 REPORTER_ASSERT(r, DSLType(kFloat2_Type).isVector());
527 REPORTER_ASSERT(r, !DSLType(kFloat2_Type).isMatrix());
528 REPORTER_ASSERT(r, !DSLType(kFloat2_Type).isArray());
529 REPORTER_ASSERT(r, !DSLType(kFloat2_Type).isStruct());
530
531 REPORTER_ASSERT(r, !DSLType(kHalf2x2_Type).isBoolean());
532 REPORTER_ASSERT(r, !DSLType(kHalf2x2_Type).isNumber());
533 REPORTER_ASSERT(r, !DSLType(kHalf2x2_Type).isFloat());
534 REPORTER_ASSERT(r, !DSLType(kHalf2x2_Type).isSigned());
535 REPORTER_ASSERT(r, !DSLType(kHalf2x2_Type).isUnsigned());
536 REPORTER_ASSERT(r, !DSLType(kHalf2x2_Type).isInteger());
537 REPORTER_ASSERT(r, !DSLType(kHalf2x2_Type).isScalar());
538 REPORTER_ASSERT(r, !DSLType(kHalf2x2_Type).isVector());
539 REPORTER_ASSERT(r, DSLType(kHalf2x2_Type).isMatrix());
540 REPORTER_ASSERT(r, !DSLType(kHalf2x2_Type).isArray());
541 REPORTER_ASSERT(r, !DSLType(kHalf2x2_Type).isStruct());
542
543 REPORTER_ASSERT(r, !DSLType(Array(kFloat_Type, 2)).isBoolean());
544 REPORTER_ASSERT(r, !DSLType(Array(kFloat_Type, 2)).isNumber());
545 REPORTER_ASSERT(r, !DSLType(Array(kFloat_Type, 2)).isFloat());
546 REPORTER_ASSERT(r, !DSLType(Array(kFloat_Type, 2)).isSigned());
547 REPORTER_ASSERT(r, !DSLType(Array(kFloat_Type, 2)).isUnsigned());
548 REPORTER_ASSERT(r, !DSLType(Array(kFloat_Type, 2)).isInteger());
549 REPORTER_ASSERT(r, !DSLType(Array(kFloat_Type, 2)).isScalar());
550 REPORTER_ASSERT(r, !DSLType(Array(kFloat_Type, 2)).isVector());
551 REPORTER_ASSERT(r, !DSLType(Array(kFloat_Type, 2)).isMatrix());
552 REPORTER_ASSERT(r, DSLType(Array(kFloat_Type, 2)).isArray());
553 REPORTER_ASSERT(r, !DSLType(Array(kFloat_Type, 2)).isStruct());
Ethan Nicholas722cb672021-05-06 10:47:06 -0400554
555 Var x(kFloat_Type);
556 DSLExpression e = x + 1;
557 REPORTER_ASSERT(r, e.type().isFloat());
558 e.release();
Ethan Nicholased6e54f2021-07-19 13:33:47 -0400559
560 {
561 ExpectError error(r, "error: array size must be positive\n");
562 Array(kFloat_Type, -1);
563 }
564
565 {
566 ExpectError error(r, "error: multidimensional arrays are not permitted\n");
567 Array(Array(kFloat_Type, 2), 2);
568 }
Ethan Nicholasb83199e2021-05-03 14:25:35 -0400569}
570
Ethan Nicholas84558932021-04-12 16:56:37 -0400571DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLMatrices, r, ctxInfo) {
572 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
573 Var f22(kFloat2x2_Type, "f22");
574 EXPECT_EQUAL(f22 = Float2x2(1), "(f22 = float2x2(1.0))");
575 Var f32(kFloat3x2_Type, "f32");
576 EXPECT_EQUAL(f32 = Float3x2(1, 2, 3, 4, 5, 6),
577 "(f32 = float3x2(1.0, 2.0, 3.0, 4.0, 5.0, 6.0))");
578 Var f42(kFloat4x2_Type, "f42");
579 EXPECT_EQUAL(f42 = Float4x2(Float4(1, 2, 3, 4), 5, 6, 7, 8),
580 "(f42 = float4x2(1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0))");
581 Var f23(kFloat2x3_Type, "f23");
582 EXPECT_EQUAL(f23 = Float2x3(1, Float2(2, 3), 4, Float2(5, 6)),
583 "(f23 = float2x3(1.0, 2.0, 3.0, 4.0, 5.0, 6.0))");
584 Var f33(kFloat3x3_Type, "f33");
585 EXPECT_EQUAL(f33 = Float3x3(Float3(1, 2, 3), 4, Float2(5, 6), 7, 8, 9),
586 "(f33 = float3x3(1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0))");
587 Var f43(kFloat4x3_Type, "f43");
588 EXPECT_EQUAL(f43 = Float4x3(Float4(1, 2, 3, 4), Float4(5, 6, 7, 8), Float4(9, 10, 11, 12)),
589 "(f43 = float4x3(1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0, 10.0, 11.0, 12.0))");
590 Var f24(kFloat2x4_Type, "f24");
591 EXPECT_EQUAL(f24 = Float2x4(1, 2, 3, 4, 5, 6, 7, 8),
592 "(f24 = float2x4(1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0))");
593 Var f34(kFloat3x4_Type, "f34");
594 EXPECT_EQUAL(f34 = Float3x4(1, 2, 3, 4, 5, 6, 7, 8, 9, Float3(10, 11, 12)),
595 "(f34 = float3x4(1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0, 10.0, 11.0, 12.0))");
596 Var f44(kFloat4x4_Type, "f44");
597 EXPECT_EQUAL(f44 = Float4x4(1), "(f44 = float4x4(1.0))");
598
599 Var h22(kHalf2x2_Type, "h22");
600 EXPECT_EQUAL(h22 = Half2x2(1), "(h22 = half2x2(1.0))");
601 Var h32(kHalf3x2_Type, "h32");
602 EXPECT_EQUAL(h32 = Half3x2(1, 2, 3, 4, 5, 6),
603 "(h32 = half3x2(1.0, 2.0, 3.0, 4.0, 5.0, 6.0))");
604 Var h42(kHalf4x2_Type, "h42");
605 EXPECT_EQUAL(h42 = Half4x2(Half4(1, 2, 3, 4), 5, 6, 7, 8),
606 "(h42 = half4x2(1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0))");
607 Var h23(kHalf2x3_Type, "h23");
608 EXPECT_EQUAL(h23 = Half2x3(1, Half2(2, 3), 4, Half2(5, 6)),
609 "(h23 = half2x3(1.0, 2.0, 3.0, 4.0, 5.0, 6.0))");
610 Var h33(kHalf3x3_Type, "h33");
611 EXPECT_EQUAL(h33 = Half3x3(Half3(1, 2, 3), 4, Half2(5, 6), 7, 8, 9),
612 "(h33 = half3x3(1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0))");
613 Var h43(kHalf4x3_Type, "h43");
614 EXPECT_EQUAL(h43 = Half4x3(Half4(1, 2, 3, 4), Half4(5, 6, 7, 8), Half4(9, 10, 11, 12)),
615 "(h43 = half4x3(1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0, 10.0, 11.0, 12.0))");
616 Var h24(kHalf2x4_Type, "h24");
617 EXPECT_EQUAL(h24 = Half2x4(1, 2, 3, 4, 5, 6, 7, 8),
618 "(h24 = half2x4(1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0))");
619 Var h34(kHalf3x4_Type, "h34");
620 EXPECT_EQUAL(h34 = Half3x4(1, 2, 3, 4, 5, 6, 7, 8, 9, Half3(10, 11, 12)),
621 "(h34 = half3x4(1.0, 2.0, 3.0, 4.0, 5.0, 6.0, 7.0, 8.0, 9.0, 10.0, 11.0, 12.0))");
622 Var h44(kHalf4x4_Type, "h44");
623 EXPECT_EQUAL(h44 = Half4x4(1), "(h44 = half4x4(1.0))");
624
625 EXPECT_EQUAL(f22 * 2, "(f22 * 2.0)");
626 EXPECT_EQUAL(f22 == Float2x2(1), "(f22 == float2x2(1.0))");
627 EXPECT_EQUAL(h42[0][1], "h42[0].y");
628 EXPECT_EQUAL(f43 * Float4(0), "(f43 * float4(0.0))");
629 EXPECT_EQUAL(h23 * 2, "(h23 * 2.0)");
630 EXPECT_EQUAL(Inverse(f44), "inverse(f44)");
631
632 {
633 ExpectError error(r, "error: invalid arguments to 'float3x3' constructor (expected 9 "
634 "scalars, but found 2)\n");
635 DSLExpression(Float3x3(Float2(1))).release();
636 }
637
638 {
639 ExpectError error(r, "error: invalid arguments to 'half2x2' constructor (expected 4 "
640 "scalars, but found 5)\n");
641 DSLExpression(Half2x2(1, 2, 3, 4, 5)).release();
642 }
643
644 {
645 ExpectError error(r, "error: type mismatch: '*' cannot operate on 'float4x3', 'float3'\n");
646 DSLExpression(f43 * Float3(1)).release();
647 }
648
649 {
650 ExpectError error(r, "error: type mismatch: '=' cannot operate on 'float4x3', "
651 "'float3x3'\n");
652 DSLExpression(f43 = f33).release();
653 }
654
655 {
656 ExpectError error(r, "error: type mismatch: '=' cannot operate on 'half2x2', "
657 "'float2x2'\n");
658 DSLExpression(h22 = f22).release();
659 }
660
661 {
662 ExpectError error(r,
663 "error: no match for inverse(float4x3)\n");
664 DSLExpression(Inverse(f43)).release();
665 }
666}
667
Ethan Nicholas92969f22021-01-13 10:38:59 -0500668DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLPlus, r, ctxInfo) {
669 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -0400670 Var a(kFloat_Type, "a"), b(kFloat_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500671
John Stiles8f440b42021-03-05 16:48:56 -0500672 EXPECT_EQUAL(a + b,
673 "(a + b)");
674 EXPECT_EQUAL(a + 1,
675 "(a + 1.0)");
676 EXPECT_EQUAL(0.5 + a + -99,
677 "((0.5 + a) + -99.0)");
678 EXPECT_EQUAL(a += b + 1,
679 "(a += (b + 1.0))");
Ethan Nicholasb14b6362021-03-08 17:07:58 -0500680 EXPECT_EQUAL(+a,
681 "a");
682 EXPECT_EQUAL(+(a + b),
683 "(a + b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500684
685 {
686 ExpectError error(r, "error: type mismatch: '+' cannot operate on 'bool2', 'float'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500687 DSLExpression((Bool2(true) + a)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500688 }
689
690 {
691 ExpectError error(r, "error: type mismatch: '+=' cannot operate on 'float', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500692 DSLExpression((a += Bool2(true))).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500693 }
694
695 {
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500696 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500697 DSLExpression((1.0 += a)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500698 }
Ethan Nicholasb14b6362021-03-08 17:07:58 -0500699
700 {
701 ExpectError error(r, "error: '+' cannot operate on 'bool'\n");
Ethan Nicholasb14e6b92021-04-08 16:56:05 -0400702 Var c(kBool_Type);
Ethan Nicholas549c6b82021-06-25 12:31:44 -0400703 DSLExpression(+c).release();
Ethan Nicholasb14b6362021-03-08 17:07:58 -0500704 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500705}
706
707DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLMinus, r, ctxInfo) {
708 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -0400709 Var a(kInt_Type, "a"), b(kInt_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500710
John Stiles8f440b42021-03-05 16:48:56 -0500711 EXPECT_EQUAL(a - b,
712 "(a - b)");
713 EXPECT_EQUAL(a - 1,
714 "(a - 1)");
715 EXPECT_EQUAL(2 - a - b,
716 "((2 - a) - b)");
717 EXPECT_EQUAL(a -= b + 1,
718 "(a -= (b + 1))");
Ethan Nicholasb14b6362021-03-08 17:07:58 -0500719 EXPECT_EQUAL(-a,
720 "-a");
721 EXPECT_EQUAL(-(a - b),
722 "-(a - b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500723
724 {
725 ExpectError error(r, "error: type mismatch: '-' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500726 DSLExpression(Bool2(true) - a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500727 }
728
729 {
730 ExpectError error(r, "error: type mismatch: '-=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500731 DSLExpression(a -= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500732 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500733
734 {
735 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500736 DSLExpression(1.0 -= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500737 }
Ethan Nicholasb14b6362021-03-08 17:07:58 -0500738
739 {
740 ExpectError error(r, "error: '-' cannot operate on 'bool'\n");
Ethan Nicholasb14e6b92021-04-08 16:56:05 -0400741 Var c(kBool_Type);
Ethan Nicholas549c6b82021-06-25 12:31:44 -0400742 DSLExpression(-c).release();
Ethan Nicholasb14b6362021-03-08 17:07:58 -0500743 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500744}
745
746DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLMultiply, r, ctxInfo) {
747 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -0400748 Var a(kFloat_Type, "a"), b(kFloat_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500749
John Stiles8f440b42021-03-05 16:48:56 -0500750 EXPECT_EQUAL(a * b,
751 "(a * b)");
752 EXPECT_EQUAL(a * 2,
753 "(a * 2.0)");
754 EXPECT_EQUAL(0.5 * a * -99,
755 "((0.5 * a) * -99.0)");
756 EXPECT_EQUAL(a *= b + 1,
757 "(a *= (b + 1.0))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500758
759 {
760 ExpectError error(r, "error: type mismatch: '*' cannot operate on 'bool2', 'float'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500761 DSLExpression(Bool2(true) * a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500762 }
763
764 {
765 ExpectError error(r, "error: type mismatch: '*=' cannot operate on 'float', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500766 DSLExpression(a *= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500767 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500768
769 {
770 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500771 DSLExpression(1.0 *= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500772 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500773}
774
775DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDivide, r, ctxInfo) {
776 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -0400777 Var a(kFloat_Type, "a"), b(kFloat_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500778
John Stiles8f440b42021-03-05 16:48:56 -0500779 EXPECT_EQUAL(a / b,
780 "(a / b)");
781 EXPECT_EQUAL(a / 2,
782 "(a / 2.0)");
783 EXPECT_EQUAL(0.5 / a / -99,
784 "((0.5 / a) / -99.0)");
785 EXPECT_EQUAL(b / (a - 1),
786 "(b / (a - 1.0))");
787 EXPECT_EQUAL(a /= b + 1,
788 "(a /= (b + 1.0))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500789
790 {
791 ExpectError error(r, "error: type mismatch: '/' cannot operate on 'bool2', 'float'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500792 DSLExpression(Bool2(true) / a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500793 }
794
795 {
796 ExpectError error(r, "error: type mismatch: '/=' cannot operate on 'float', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500797 DSLExpression(a /= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500798 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500799
800 {
801 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500802 DSLExpression(1.0 /= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500803 }
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500804
805 {
806 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500807 DSLExpression(a /= 0).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500808 }
809
810 {
Ethan Nicholasb14e6b92021-04-08 16:56:05 -0400811 Var c(kFloat2_Type, "c");
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500812 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500813 DSLExpression(c /= Float2(Float(0), 1)).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500814 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500815}
816
817DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLMod, r, ctxInfo) {
818 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -0400819 Var a(kInt_Type, "a"), b(kInt_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500820 Expression e1 = a % b;
John Stilesb4d7b582021-02-19 09:56:31 -0500821 EXPECT_EQUAL(e1, "(a % b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500822
823 Expression e2 = a % 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500824 EXPECT_EQUAL(e2, "(a % 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500825
826 Expression e3 = 10 % a % -99;
John Stilesb4d7b582021-02-19 09:56:31 -0500827 EXPECT_EQUAL(e3, "((10 % a) % -99)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500828
829 Expression e4 = a %= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500830 EXPECT_EQUAL(e4, "(a %= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500831
832 {
833 ExpectError error(r, "error: type mismatch: '%' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500834 DSLExpression(Bool2(true) % a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500835 }
836
837 {
838 ExpectError error(r, "error: type mismatch: '%=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500839 DSLExpression(a %= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500840 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500841
842 {
843 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500844 DSLExpression(1 %= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500845 }
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500846
847 {
848 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500849 DSLExpression(a %= 0).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500850 }
851
852 {
Ethan Nicholasb14e6b92021-04-08 16:56:05 -0400853 Var c(kInt2_Type, "c");
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500854 ExpectError error(r, "error: division by zero\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500855 DSLExpression(c %= Int2(Int(0), 1)).release();
Ethan Nicholasc0f98152021-02-05 16:21:10 -0500856 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500857}
858
859DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLShl, r, ctxInfo) {
860 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -0400861 Var a(kInt_Type, "a"), b(kInt_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500862 Expression e1 = a << b;
John Stilesb4d7b582021-02-19 09:56:31 -0500863 EXPECT_EQUAL(e1, "(a << b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500864
865 Expression e2 = a << 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500866 EXPECT_EQUAL(e2, "(a << 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500867
868 Expression e3 = 1 << a << 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500869 EXPECT_EQUAL(e3, "((1 << a) << 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500870
871 Expression e4 = a <<= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500872 EXPECT_EQUAL(e4, "(a <<= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500873
874 {
875 ExpectError error(r, "error: type mismatch: '<<' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500876 DSLExpression(Bool2(true) << a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500877 }
878
879 {
880 ExpectError error(r, "error: type mismatch: '<<=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500881 DSLExpression(a <<= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500882 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500883
884 {
885 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500886 DSLExpression(1 <<= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500887 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500888}
889
890DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLShr, r, ctxInfo) {
891 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -0400892 Var a(kInt_Type, "a"), b(kInt_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500893 Expression e1 = a >> b;
John Stilesb4d7b582021-02-19 09:56:31 -0500894 EXPECT_EQUAL(e1, "(a >> b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500895
896 Expression e2 = a >> 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500897 EXPECT_EQUAL(e2, "(a >> 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500898
899 Expression e3 = 1 >> a >> 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500900 EXPECT_EQUAL(e3, "((1 >> a) >> 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500901
902 Expression e4 = a >>= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500903 EXPECT_EQUAL(e4, "(a >>= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500904
905 {
906 ExpectError error(r, "error: type mismatch: '>>' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500907 DSLExpression(Bool2(true) >> a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500908 }
909
910 {
911 ExpectError error(r, "error: type mismatch: '>>=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500912 DSLExpression(a >>= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500913 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500914
915 {
916 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500917 DSLExpression(1 >>= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500918 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500919}
920
921DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseAnd, r, ctxInfo) {
922 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -0400923 Var a(kInt_Type, "a"), b(kInt_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500924 Expression e1 = a & b;
John Stilesb4d7b582021-02-19 09:56:31 -0500925 EXPECT_EQUAL(e1, "(a & b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500926
927 Expression e2 = a & 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500928 EXPECT_EQUAL(e2, "(a & 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500929
930 Expression e3 = 1 & a & 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500931 EXPECT_EQUAL(e3, "((1 & a) & 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500932
933 Expression e4 = a &= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500934 EXPECT_EQUAL(e4, "(a &= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500935
936 {
937 ExpectError error(r, "error: type mismatch: '&' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500938 DSLExpression(Bool2(true) & a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500939 }
940
941 {
942 ExpectError error(r, "error: type mismatch: '&=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500943 DSLExpression(a &= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500944 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500945
946 {
947 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500948 DSLExpression(1 &= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500949 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500950}
951
952DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseOr, r, ctxInfo) {
953 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -0400954 Var a(kInt_Type, "a"), b(kInt_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500955 Expression e1 = a | b;
John Stilesb4d7b582021-02-19 09:56:31 -0500956 EXPECT_EQUAL(e1, "(a | b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500957
958 Expression e2 = a | 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500959 EXPECT_EQUAL(e2, "(a | 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500960
961 Expression e3 = 1 | a | 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500962 EXPECT_EQUAL(e3, "((1 | a) | 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500963
964 Expression e4 = a |= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500965 EXPECT_EQUAL(e4, "(a |= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500966
967 {
968 ExpectError error(r, "error: type mismatch: '|' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500969 DSLExpression(Bool2(true) | a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500970 }
971
972 {
973 ExpectError error(r, "error: type mismatch: '|=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500974 DSLExpression(a |= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -0500975 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500976
977 {
978 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -0500979 DSLExpression(1 |= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -0500980 }
Ethan Nicholas92969f22021-01-13 10:38:59 -0500981}
982
983DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseXor, r, ctxInfo) {
984 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -0400985 Var a(kInt_Type, "a"), b(kInt_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500986 Expression e1 = a ^ b;
John Stilesb4d7b582021-02-19 09:56:31 -0500987 EXPECT_EQUAL(e1, "(a ^ b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500988
989 Expression e2 = a ^ 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500990 EXPECT_EQUAL(e2, "(a ^ 1)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500991
992 Expression e3 = 1 ^ a ^ 2;
John Stilesb4d7b582021-02-19 09:56:31 -0500993 EXPECT_EQUAL(e3, "((1 ^ a) ^ 2)");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500994
995 Expression e4 = a ^= b + 1;
John Stilesb4d7b582021-02-19 09:56:31 -0500996 EXPECT_EQUAL(e4, "(a ^= (b + 1))");
Ethan Nicholas92969f22021-01-13 10:38:59 -0500997
998 {
999 ExpectError error(r, "error: type mismatch: '^' cannot operate on 'bool2', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001000 DSLExpression(Bool2(true) ^ a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001001 }
1002
1003 {
1004 ExpectError error(r, "error: type mismatch: '^=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001005 DSLExpression(a ^= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001006 }
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -05001007
1008 {
1009 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001010 DSLExpression(1 ^= a).release();
Ethan Nicholas67a0a8a2021-01-13 12:36:02 -05001011 }
Ethan Nicholas92969f22021-01-13 10:38:59 -05001012}
1013
1014DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLogicalAnd, r, ctxInfo) {
1015 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001016 Var a(kBool_Type, "a"), b(kBool_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001017 Expression e1 = a && b;
John Stilesb4d7b582021-02-19 09:56:31 -05001018 EXPECT_EQUAL(e1, "(a && b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001019
1020 Expression e2 = a && true && b;
John Stilesb4d7b582021-02-19 09:56:31 -05001021 EXPECT_EQUAL(e2, "(a && b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001022
1023 Expression e3 = a && false && b;
John Stilesb4d7b582021-02-19 09:56:31 -05001024 EXPECT_EQUAL(e3, "false");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001025
1026 {
1027 ExpectError error(r, "error: type mismatch: '&&' cannot operate on 'bool', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001028 DSLExpression(a && 5).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001029 }
1030}
1031
1032DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLogicalOr, r, ctxInfo) {
1033 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001034 Var a(kBool_Type, "a"), b(kBool_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001035 Expression e1 = a || b;
John Stilesb4d7b582021-02-19 09:56:31 -05001036 EXPECT_EQUAL(e1, "(a || b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001037
1038 Expression e2 = a || true || b;
John Stilesb4d7b582021-02-19 09:56:31 -05001039 EXPECT_EQUAL(e2, "true");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001040
1041 Expression e3 = a || false || b;
John Stilesb4d7b582021-02-19 09:56:31 -05001042 EXPECT_EQUAL(e3, "(a || b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001043
1044 {
1045 ExpectError error(r, "error: type mismatch: '||' cannot operate on 'bool', 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001046 DSLExpression(a || 5).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001047 }
1048}
1049
Ethan Nicholas2ab47c92021-07-19 12:30:37 -04001050DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLogicalXor, r, ctxInfo) {
1051 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1052 Var a(kBool_Type, "a"), b(kBool_Type, "b");
1053 Expression e1 = LogicalXor(a, b);
1054 EXPECT_EQUAL(e1, "(a ^^ b)");
1055
1056 {
1057 ExpectError error(r, "error: type mismatch: '^^' cannot operate on 'bool', 'int'\n");
1058 DSLExpression(LogicalXor(a, 5)).release();
1059 }
1060}
1061
Ethan Nicholas92969f22021-01-13 10:38:59 -05001062DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLComma, r, ctxInfo) {
1063 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001064 Var a(kInt_Type, "a"), b(kInt_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001065 Expression e1 = (a += b, b);
John Stilesb4d7b582021-02-19 09:56:31 -05001066 EXPECT_EQUAL(e1, "((a += b) , b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001067
1068 Expression e2 = (a += b, b += b, Int2(a));
John Stilesb4d7b582021-02-19 09:56:31 -05001069 EXPECT_EQUAL(e2, "(((a += b) , (b += b)) , int2(a))");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001070}
1071
1072DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLEqual, r, ctxInfo) {
1073 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001074 Var a(kInt_Type, "a"), b(kInt_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001075 Expression e1 = a == b;
John Stilesb4d7b582021-02-19 09:56:31 -05001076 EXPECT_EQUAL(e1, "(a == b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001077
1078 Expression e2 = a == 5;
John Stilesb4d7b582021-02-19 09:56:31 -05001079 EXPECT_EQUAL(e2, "(a == 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001080
1081 {
1082 ExpectError error(r, "error: type mismatch: '==' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001083 DSLExpression(a == Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001084 }
1085}
1086
1087DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLNotEqual, r, ctxInfo) {
1088 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001089 Var a(kInt_Type, "a"), b(kInt_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001090 Expression e1 = a != b;
John Stilesb4d7b582021-02-19 09:56:31 -05001091 EXPECT_EQUAL(e1, "(a != b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001092
1093 Expression e2 = a != 5;
John Stilesb4d7b582021-02-19 09:56:31 -05001094 EXPECT_EQUAL(e2, "(a != 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001095
1096 {
1097 ExpectError error(r, "error: type mismatch: '!=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001098 DSLExpression(a != Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001099 }
1100}
1101
1102DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLGreaterThan, r, ctxInfo) {
1103 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001104 Var a(kInt_Type, "a"), b(kInt_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001105 Expression e1 = a > b;
John Stilesb4d7b582021-02-19 09:56:31 -05001106 EXPECT_EQUAL(e1, "(a > b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001107
1108 Expression e2 = a > 5;
John Stilesb4d7b582021-02-19 09:56:31 -05001109 EXPECT_EQUAL(e2, "(a > 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001110
1111 {
1112 ExpectError error(r, "error: type mismatch: '>' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001113 DSLExpression(a > Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001114 }
1115}
1116
1117DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLGreaterThanOrEqual, r, ctxInfo) {
1118 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001119 Var a(kInt_Type, "a"), b(kInt_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001120 Expression e1 = a >= b;
John Stilesb4d7b582021-02-19 09:56:31 -05001121 EXPECT_EQUAL(e1, "(a >= b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001122
1123 Expression e2 = a >= 5;
John Stilesb4d7b582021-02-19 09:56:31 -05001124 EXPECT_EQUAL(e2, "(a >= 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001125
1126 {
1127 ExpectError error(r, "error: type mismatch: '>=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001128 DSLExpression(a >= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001129 }
1130}
1131
1132DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLessThan, r, ctxInfo) {
1133 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001134 Var a(kInt_Type, "a"), b(kInt_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001135 Expression e1 = a < b;
John Stilesb4d7b582021-02-19 09:56:31 -05001136 EXPECT_EQUAL(e1, "(a < b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001137
1138 Expression e2 = a < 5;
John Stilesb4d7b582021-02-19 09:56:31 -05001139 EXPECT_EQUAL(e2, "(a < 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001140
1141 {
1142 ExpectError error(r, "error: type mismatch: '<' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001143 DSLExpression(a < Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001144 }
1145}
1146
1147DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLessThanOrEqual, r, ctxInfo) {
1148 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001149 Var a(kInt_Type, "a"), b(kInt_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001150 Expression e1 = a <= b;
John Stilesb4d7b582021-02-19 09:56:31 -05001151 EXPECT_EQUAL(e1, "(a <= b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001152
1153 Expression e2 = a <= 5;
John Stilesb4d7b582021-02-19 09:56:31 -05001154 EXPECT_EQUAL(e2, "(a <= 5)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001155
1156 {
1157 ExpectError error(r, "error: type mismatch: '<=' cannot operate on 'int', 'bool2'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001158 DSLExpression(a <= Bool2(true)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001159 }
1160}
1161
1162DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLogicalNot, r, ctxInfo) {
1163 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001164 Var a(kInt_Type, "a"), b(kInt_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001165 Expression e1 = !(a <= b);
John Stilesb4d7b582021-02-19 09:56:31 -05001166 EXPECT_EQUAL(e1, "!(a <= b)");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001167
1168 {
1169 ExpectError error(r, "error: '!' cannot operate on 'int'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001170 DSLExpression(!a).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001171 }
1172}
1173
1174DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBitwiseNot, r, ctxInfo) {
1175 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001176 Var a(kInt_Type, "a"), b(kBool_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001177 Expression e1 = ~a;
John Stilesb4d7b582021-02-19 09:56:31 -05001178 EXPECT_EQUAL(e1, "~a");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001179
1180 {
1181 ExpectError error(r, "error: '~' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001182 DSLExpression(~b).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001183 }
1184}
1185
1186DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLIncrement, r, ctxInfo) {
1187 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001188 Var a(kInt_Type, "a"), b(kBool_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001189 Expression e1 = ++a;
John Stilesb4d7b582021-02-19 09:56:31 -05001190 EXPECT_EQUAL(e1, "++a");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001191
1192 Expression e2 = a++;
John Stilesb4d7b582021-02-19 09:56:31 -05001193 EXPECT_EQUAL(e2, "a++");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001194
1195 {
1196 ExpectError error(r, "error: '++' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001197 DSLExpression(++b).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001198 }
1199
1200 {
1201 ExpectError error(r, "error: '++' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001202 DSLExpression(b++).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001203 }
1204
1205 {
1206 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001207 DSLExpression(++(a + 1)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001208 }
1209
1210 {
1211 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001212 DSLExpression((a + 1)++).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001213 }
1214}
1215
1216DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDecrement, r, ctxInfo) {
1217 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001218 Var a(kInt_Type, "a"), b(kBool_Type, "b");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001219 Expression e1 = --a;
John Stilesb4d7b582021-02-19 09:56:31 -05001220 EXPECT_EQUAL(e1, "--a");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001221
1222 Expression e2 = a--;
John Stilesb4d7b582021-02-19 09:56:31 -05001223 EXPECT_EQUAL(e2, "a--");
Ethan Nicholas92969f22021-01-13 10:38:59 -05001224
1225 {
1226 ExpectError error(r, "error: '--' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001227 DSLExpression(--b).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001228 }
1229
1230 {
1231 ExpectError error(r, "error: '--' cannot operate on 'bool'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001232 DSLExpression(b--).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001233 }
1234
1235 {
1236 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001237 DSLExpression(--(a + 1)).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001238 }
1239
1240 {
1241 ExpectError error(r, "error: cannot assign to this expression\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001242 DSLExpression((a + 1)--).release();
Ethan Nicholas92969f22021-01-13 10:38:59 -05001243 }
1244}
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001245
Ethan Nicholas722cb672021-05-06 10:47:06 -04001246DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLCall, r, ctxInfo) {
1247 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1248 {
Ethan Nicholasb4f8b7a2021-06-23 10:27:09 -04001249 DSLExpression sqrt(DSLWriter::IRGenerator().convertIdentifier(/*offset=*/-1, "sqrt"));
Ethan Nicholas722cb672021-05-06 10:47:06 -04001250 SkTArray<DSLWrapper<DSLExpression>> args;
John Stiles443fa192021-05-24 23:46:46 -04001251 args.emplace_back(16);
1252 EXPECT_EQUAL(sqrt(std::move(args)), "4.0"); // sqrt(16) gets optimized to 4
Ethan Nicholas722cb672021-05-06 10:47:06 -04001253 }
1254
1255 {
Ethan Nicholasb4f8b7a2021-06-23 10:27:09 -04001256 DSLExpression pow(DSLWriter::IRGenerator().convertIdentifier(/*offset=*/-1, "pow"));
Ethan Nicholas722cb672021-05-06 10:47:06 -04001257 DSLVar a(kFloat_Type, "a");
1258 DSLVar b(kFloat_Type, "b");
1259 SkTArray<DSLWrapper<DSLExpression>> args;
1260 args.emplace_back(a);
1261 args.emplace_back(b);
1262 EXPECT_EQUAL(pow(std::move(args)), "pow(a, b)");
1263 }
1264}
1265
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001266DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBlock, r, ctxInfo) {
Ethan Nicholas55a63af2021-05-18 10:12:58 -04001267 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), no_mark_vars_declared());
Ethan Nicholas722cb672021-05-06 10:47:06 -04001268 EXPECT_EQUAL(Block(), "{ }");
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001269 Var a(kInt_Type, "a", 1), b(kInt_Type, "b", 2);
Ethan Nicholas722cb672021-05-06 10:47:06 -04001270 EXPECT_EQUAL(Block(Declare(a), Declare(b), a = b), "{ int a = 1; int b = 2; (a = b); }");
Ethan Nicholasdb2326b2021-04-19 10:55:18 -04001271
Ethan Nicholas722cb672021-05-06 10:47:06 -04001272 EXPECT_EQUAL((If(a > 0, --a), ++b), "if ((a > 0)) --a; ++b;");
1273
1274 SkTArray<DSLStatement> statements;
1275 statements.push_back(a = 0);
1276 statements.push_back(++a);
1277 EXPECT_EQUAL(Block(std::move(statements)), "{ (a = 0); ++a; }");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001278}
1279
Ethan Nicholasdaceb792021-02-05 14:22:32 -05001280DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBreak, r, ctxInfo) {
Ethan Nicholas55a63af2021-05-18 10:12:58 -04001281 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), no_mark_vars_declared());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001282 Var i(kInt_Type, "i", 0);
1283 DSLFunction(kVoid_Type, "success").define(
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001284 For(Declare(i), i < 10, ++i, Block(
Ethan Nicholasdaceb792021-02-05 14:22:32 -05001285 If(i > 5, Break())
1286 ))
1287 );
1288 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
John Stilesb4d7b582021-02-19 09:56:31 -05001289 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1290 "void success() { for (int i = 0; (i < 10); ++i) { if ((i > 5)) break; } }");
Ethan Nicholasdaceb792021-02-05 14:22:32 -05001291
1292 {
1293 ExpectError error(r, "error: break statement must be inside a loop or switch\n");
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001294 DSLFunction(kVoid_Type, "fail").define(
Ethan Nicholasdaceb792021-02-05 14:22:32 -05001295 Break()
1296 );
1297 }
1298}
1299
1300DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLContinue, r, ctxInfo) {
Ethan Nicholas55a63af2021-05-18 10:12:58 -04001301 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), no_mark_vars_declared());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001302 Var i(kInt_Type, "i", 0);
1303 DSLFunction(kVoid_Type, "success").define(
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001304 For(Declare(i), i < 10, ++i, Block(
Ethan Nicholasdaceb792021-02-05 14:22:32 -05001305 If(i < 5, Continue())
1306 ))
1307 );
1308 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
John Stilesb4d7b582021-02-19 09:56:31 -05001309 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1310 "void success() { for (int i = 0; (i < 10); ++i) { if ((i < 5)) continue; } }");
Ethan Nicholasdaceb792021-02-05 14:22:32 -05001311
1312 {
1313 ExpectError error(r, "error: continue statement must be inside a loop\n");
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001314 DSLFunction(kVoid_Type, "fail").define(
Ethan Nicholasdaceb792021-02-05 14:22:32 -05001315 Continue()
1316 );
1317 }
1318}
1319
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001320DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDeclare, r, ctxInfo) {
Ethan Nicholas55a63af2021-05-18 10:12:58 -04001321 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), no_mark_vars_declared());
Ethan Nicholasda729782021-07-19 12:55:52 -04001322 {
1323 Var a(kHalf4_Type, "a"), b(kHalf4_Type, "b", Half4(1));
1324 EXPECT_EQUAL(Declare(a), "half4 a;");
1325 EXPECT_EQUAL(Declare(b), "half4 b = half4(1.0);");
1326 }
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001327
1328 {
Ethan Nicholasda729782021-07-19 12:55:52 -04001329 DSLWriter::Reset();
1330 SkTArray<Var> vars;
1331 vars.push_back(Var(kBool_Type, "a", true));
1332 vars.push_back(Var(kFloat_Type, "b"));
1333 EXPECT_EQUAL(Declare(vars), "bool a = true; float b;");
1334 }
1335
1336 {
1337 DSLWriter::Reset();
1338 REPORTER_ASSERT(r, DSLWriter::ProgramElements().empty());
1339 GlobalVar a(kHalf4_Type, "a"), b(kHalf4_Type, "b", Half4(1));
1340 Declare(a);
1341 Declare(b);
1342 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 2);
1343 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0], "half4 a;");
1344 EXPECT_EQUAL(*DSLWriter::ProgramElements()[1], "half4 b = half4(1.0);");
1345 }
1346
1347 {
1348 DSLWriter::Reset();
1349 REPORTER_ASSERT(r, DSLWriter::ProgramElements().empty());
1350 SkTArray<GlobalVar> vars;
1351 vars.push_back(GlobalVar(kHalf4_Type, "a"));
1352 vars.push_back(GlobalVar(kHalf4_Type, "b", Half4(1)));
1353 Declare(vars);
1354 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 2);
1355 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0], "half4 a;");
1356 EXPECT_EQUAL(*DSLWriter::ProgramElements()[1], "half4 b = half4(1.0);");
1357 }
1358
1359 {
1360 DSLWriter::Reset();
1361 Var a(kHalf4_Type, "a", 1);
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001362 ExpectError error(r, "error: expected 'half4', but found 'int'\n");
Ethan Nicholasda729782021-07-19 12:55:52 -04001363 Declare(a).release();
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001364 }
1365
1366 {
Ethan Nicholasda729782021-07-19 12:55:52 -04001367 DSLWriter::Reset();
1368 Var a(kInt_Type, "a");
1369 Declare(a).release();
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001370 ExpectError error(r, "error: variable has already been declared\n");
Ethan Nicholasda729782021-07-19 12:55:52 -04001371 Declare(a).release();
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001372 }
Ethan Nicholase9c2c5a2021-04-30 13:14:24 -04001373
1374 {
Ethan Nicholasda729782021-07-19 12:55:52 -04001375 DSLWriter::Reset();
1376 Var a(kUniform_Modifier, kInt_Type, "a");
Ethan Nicholasa2d22b22021-07-15 10:35:54 -04001377 ExpectError error(r, "error: 'uniform' is not permitted here\n");
Ethan Nicholasda729782021-07-19 12:55:52 -04001378 Declare(a).release();
Ethan Nicholase9c2c5a2021-04-30 13:14:24 -04001379 }
1380}
1381
1382DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDeclareGlobal, r, ctxInfo) {
Ethan Nicholas55a63af2021-05-18 10:12:58 -04001383 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), no_mark_vars_declared());
Ethan Nicholasa2d22b22021-07-15 10:35:54 -04001384 DSLGlobalVar x(kInt_Type, "x", 0);
1385 Declare(x);
1386 DSLGlobalVar y(kUniform_Modifier, kFloat2_Type, "y");
1387 Declare(y);
Ethan Nicholase9c2c5a2021-04-30 13:14:24 -04001388 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 2);
1389 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0], "int x = 0;");
1390 EXPECT_EQUAL(*DSLWriter::ProgramElements()[1], "uniform float2 y;");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001391}
1392
Ethan Nicholasdaceb792021-02-05 14:22:32 -05001393DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDiscard, r, ctxInfo) {
1394 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
John Stiles443fa192021-05-24 23:46:46 -04001395 Var x(kFloat_Type, "x", 1);
1396 EXPECT_EQUAL(If(Sqrt(x) > 0, Discard()), "if ((sqrt(x) > 0.0)) discard;");
Ethan Nicholasdaceb792021-02-05 14:22:32 -05001397}
1398
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001399DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLDo, r, ctxInfo) {
1400 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1401 Statement x = Do(Block(), true);
John Stilesb4d7b582021-02-19 09:56:31 -05001402 EXPECT_EQUAL(x, "do {} while (true);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001403
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001404 Var a(kFloat_Type, "a"), b(kFloat_Type, "b");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001405 Statement y = Do(Block(a++, --b), a != b);
John Stilesb4d7b582021-02-19 09:56:31 -05001406 EXPECT_EQUAL(y, "do { a++; --b; } while ((a != b));");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001407
1408 {
1409 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
1410 Do(Block(), 7).release();
1411 }
1412}
1413
1414DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLFor, r, ctxInfo) {
Ethan Nicholas55a63af2021-05-18 10:12:58 -04001415 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), no_mark_vars_declared());
John Stiles8676ebe2021-04-20 15:30:41 -04001416 EXPECT_EQUAL(For(Statement(), Expression(), Expression(), Block()),
1417 "for (;;) {}");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001418
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001419 Var i(kInt_Type, "i", 0);
John Stiles8676ebe2021-04-20 15:30:41 -04001420 EXPECT_EQUAL(For(Declare(i), i < 10, ++i, i += 5),
1421 "for (int i = 0; (i < 10); ++i) (i += 5);");
1422
1423 Var j(kInt_Type, "j", 0);
1424 Var k(kInt_Type, "k", 10);
1425 EXPECT_EQUAL(For((Declare(j), Declare(k)), j < k, ++j, Block()), R"(
1426 {
1427 int j = 0;
1428 int k = 10;
1429 for (; (j < k); ++j) {}
1430 }
1431 )");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001432
1433 {
1434 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
1435 For(i = 0, i + 10, ++i, i += 5).release();
1436 }
Ethan Nicholasa0f76542021-04-16 16:02:18 -04001437
1438 {
1439 ExpectError error(r, "error: invalid for loop initializer\n");
1440 For(If(i == 0, i = 1), i < 10, ++i, i += 5).release();
1441 }
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001442}
1443
Ethan Nicholase2c05042021-02-03 10:27:22 -05001444DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLFunction, r, ctxInfo) {
Ethan Nicholas55a63af2021-05-18 10:12:58 -04001445 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), no_mark_vars_declared());
Ethan Nicholasa2d22b22021-07-15 10:35:54 -04001446 Parameter coords(kFloat2_Type, "coords");
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001447 DSLFunction(kVoid_Type, "main", coords).define(
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001448 sk_FragColor() = Half4(coords, 0, 1)
1449 );
1450 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
John Stilesb4d7b582021-02-19 09:56:31 -05001451 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
Ethan Nicholas371f6e12021-05-04 14:30:02 -04001452 "void main(float2 coords) { (sk_FragColor = half4(half2(coords), 0.0, 1.0)); }");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001453
Ethan Nicholas63f75fc2021-02-23 12:05:49 -05001454 {
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001455 DSLWriter::Reset();
Ethan Nicholasa2d22b22021-07-15 10:35:54 -04001456 DSLParameter x(kFloat_Type, "x");
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001457 DSLFunction sqr(kFloat_Type, "sqr", x);
Ethan Nicholas63f75fc2021-02-23 12:05:49 -05001458 sqr.define(
1459 Return(x * x)
1460 );
1461 EXPECT_EQUAL(sqr(sk_FragCoord().x()), "sqr(sk_FragCoord.x)");
1462 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
1463 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0], "float sqr(float x) { return (x * x); }");
1464 }
1465
1466 {
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001467 DSLWriter::Reset();
Ethan Nicholasa2d22b22021-07-15 10:35:54 -04001468 DSLParameter x(kFloat2_Type, "x");
1469 DSLParameter y(kFloat2_Type, "y");
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001470 DSLFunction dot(kFloat2_Type, "dot", x, y);
Ethan Nicholas63f75fc2021-02-23 12:05:49 -05001471 dot.define(
1472 Return(x * x + y * y)
1473 );
1474 EXPECT_EQUAL(dot(Float2(1.0f, 2.0f), Float2(3.0f, 4.0f)),
1475 "dot(float2(1.0, 2.0), float2(3.0, 4.0))");
1476 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
1477 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1478 "float2 dot(float2 x, float2 y) { return ((x * x) + (y * y)); }");
1479 }
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001480
1481 {
Ethan Nicholas80f62352021-04-09 12:25:03 -04001482 DSLWriter::Reset();
Ethan Nicholasa2d22b22021-07-15 10:35:54 -04001483 DSLParameter x(kFloat_Type, "x");
1484 DSLParameter y(kFloat_Type, "y");
Ethan Nicholas80f62352021-04-09 12:25:03 -04001485 DSLFunction pair(kFloat2_Type, "pair", x, y);
1486 pair.define(
1487 Return(Float2(x, y))
1488 );
1489 Var varArg1(kFloat_Type, "varArg1");
1490 Var varArg2(kFloat_Type, "varArg2");
1491 DSLWriter::MarkDeclared(varArg1);
1492 DSLWriter::MarkDeclared(varArg2);
1493 EXPECT_EQUAL(pair(varArg1, varArg2), "pair(varArg1, varArg2)");
1494 }
1495
1496 {
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001497 ExpectError error(r, "error: expected 'float', but found 'bool'\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001498 DSLWriter::Reset();
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001499 DSLFunction(kFloat_Type, "broken").define(
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001500 Return(true)
1501 );
1502 }
1503
1504 {
John Stilesb3dcbb12021-03-04 16:00:20 -05001505 ExpectError error(r, "error: expected function to return 'float'\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001506 DSLWriter::Reset();
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001507 DSLFunction(kFloat_Type, "broken").define(
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001508 Return()
1509 );
1510 }
1511
1512 {
John Stilesb3dcbb12021-03-04 16:00:20 -05001513 ExpectError error(r, "error: function 'broken' can exit without returning a value\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001514 DSLWriter::Reset();
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001515 Var x(kFloat_Type, "x", 0);
1516 DSLFunction(kFloat_Type, "broken").define(
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001517 Declare(x),
John Stilesb3dcbb12021-03-04 16:00:20 -05001518 If(x == 1, Return(x))
1519 );
1520 }
1521
1522 {
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001523 ExpectError error(r, "error: may not return a value from a void function\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001524 DSLWriter::Reset();
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001525 DSLFunction(kVoid_Type, "broken").define(
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001526 Return(0)
1527 );
1528 }
1529
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001530 {
John Stilesb3dcbb12021-03-04 16:00:20 -05001531 ExpectError error(r, "error: function 'broken' can exit without returning a value\n");
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001532 DSLWriter::Reset();
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001533 DSLFunction(kFloat_Type, "broken").define(
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001534 );
1535 }
Ethan Nicholas961d9442021-03-16 16:37:29 -04001536
1537 {
Ethan Nicholasa2d22b22021-07-15 10:35:54 -04001538 ExpectError error(r, "error: parameter has already been used in another function\n");
Ethan Nicholas961d9442021-03-16 16:37:29 -04001539 DSLWriter::Reset();
Ethan Nicholasa2d22b22021-07-15 10:35:54 -04001540 DSLParameter p(kFloat_Type);
1541 DSLFunction(kVoid_Type, "ok", p).define(
Ethan Nicholas961d9442021-03-16 16:37:29 -04001542 );
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001543 DSLFunction(kVoid_Type, "broken", p).define(
Ethan Nicholas961d9442021-03-16 16:37:29 -04001544 );
1545 }
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001546}
1547
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001548DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLIf, r, ctxInfo) {
1549 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001550 Var a(kFloat_Type, "a"), b(kFloat_Type, "b");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001551 Statement x = If(a > b, a -= b);
John Stilesb4d7b582021-02-19 09:56:31 -05001552 EXPECT_EQUAL(x, "if ((a > b)) (a -= b);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001553
1554 Statement y = If(a > b, a -= b, b -= a);
John Stilesb4d7b582021-02-19 09:56:31 -05001555 EXPECT_EQUAL(y, "if ((a > b)) (a -= b); else (b -= a);");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001556
Ethan Nicholas8a6537d2021-04-30 12:44:00 -04001557 Statement z = StaticIf(a > b, a -= b, b -= a);
1558 EXPECT_EQUAL(z, "@if ((a > b)) (a -= b); else (b -= a);");
1559
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001560 {
1561 ExpectError error(r, "error: expected 'bool', but found 'float'\n");
1562 If(a + b, a -= b).release();
1563 }
1564}
1565
Ethan Nicholase6ed3c22021-07-08 10:38:43 -04001566DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLInterfaceBlock, r, ctxInfo) {
1567 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasa2d22b22021-07-15 10:35:54 -04001568 DSLGlobalVar intf = InterfaceBlock(kUniform_Modifier, "InterfaceBlock1",
1569 { Field(kFloat_Type, "a"), Field(kInt_Type, "b") });
Ethan Nicholase6ed3c22021-07-08 10:38:43 -04001570 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 1);
1571 EXPECT_EQUAL(*DSLWriter::ProgramElements().back(),
1572 "uniform InterfaceBlock1 { float a; int b; };");
1573 EXPECT_EQUAL(intf.field("a"), "InterfaceBlock1.a");
1574
Ethan Nicholasa2d22b22021-07-15 10:35:54 -04001575 DSLGlobalVar intf2 = InterfaceBlock(kUniform_Modifier, "InterfaceBlock2",
1576 { Field(kFloat2_Type, "x"), Field(kHalf2x2_Type, "y") },
Ethan Nicholase6ed3c22021-07-08 10:38:43 -04001577 "blockVar");
1578 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 2);
1579 EXPECT_EQUAL(*DSLWriter::ProgramElements().back(),
1580 "uniform InterfaceBlock2 { float2 x; half2x2 y; } blockVar;");
1581 EXPECT_EQUAL(intf2.field("x"), "blockVar.x");
1582
Ethan Nicholasa2d22b22021-07-15 10:35:54 -04001583 DSLGlobalVar intf3 = InterfaceBlock(kUniform_Modifier, "InterfaceBlock3",
1584 { Field(kFloat_Type, "z") },"arrayVar", 4);
Ethan Nicholase6ed3c22021-07-08 10:38:43 -04001585 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 3);
1586 EXPECT_EQUAL(*DSLWriter::ProgramElements().back(),
1587 "uniform InterfaceBlock3 { float z; } arrayVar[4];");
1588 EXPECT_EQUAL(intf3[1].field("z"), "arrayVar[1].z");
1589}
1590
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001591DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLReturn, r, ctxInfo) {
1592 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1593
1594 Statement x = Return();
John Stilesb4d7b582021-02-19 09:56:31 -05001595 EXPECT_EQUAL(x, "return;");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001596
1597 Statement y = Return(true);
John Stilesb4d7b582021-02-19 09:56:31 -05001598 EXPECT_EQUAL(y, "return true;");
Ethan Nicholas1ff76092021-01-28 10:02:43 -05001599}
1600
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001601DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSelect, r, ctxInfo) {
1602 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001603 Var a(kInt_Type, "a");
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001604 Expression x = Select(a > 0, 1, -1);
John Stilesb4d7b582021-02-19 09:56:31 -05001605 EXPECT_EQUAL(x, "((a > 0) ? 1 : -1)");
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001606
1607 {
1608 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
Ethan Nicholas549c6b82021-06-25 12:31:44 -04001609 Select(a, 1, -1).release();
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001610 }
1611
1612 {
1613 ExpectError error(r, "error: ternary operator result mismatch: 'float2', 'float3'\n");
Ethan Nicholas549c6b82021-06-25 12:31:44 -04001614 Select(a > 0, Float2(1), Float3(1)).release();
Ethan Nicholasfa648a12021-02-17 12:13:20 -05001615 }
1616}
1617
Ethan Nicholascfefec02021-02-09 15:22:57 -05001618DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSwitch, r, ctxInfo) {
1619 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1620
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001621 Var a(kFloat_Type, "a"), b(kInt_Type, "b");
Ethan Nicholascfefec02021-02-09 15:22:57 -05001622
Ethan Nicholas722cb672021-05-06 10:47:06 -04001623 SkTArray<DSLStatement> caseStatements;
1624 caseStatements.push_back(a = 1);
1625 caseStatements.push_back(Continue());
John Stilesf3a28db2021-03-10 23:00:47 -05001626 Statement x = Switch(b,
Ethan Nicholascfefec02021-02-09 15:22:57 -05001627 Case(0, a = 0, Break()),
Ethan Nicholas722cb672021-05-06 10:47:06 -04001628 Case(1, std::move(caseStatements)),
John Stilese1d1b082021-02-23 13:44:36 -05001629 Case(2, a = 2 /*Fallthrough*/),
Ethan Nicholascfefec02021-02-09 15:22:57 -05001630 Default(Discard())
1631 );
John Stilese1d1b082021-02-23 13:44:36 -05001632 EXPECT_EQUAL(x, R"(
John Stilesf3a28db2021-03-10 23:00:47 -05001633 switch (b) {
John Stilese1d1b082021-02-23 13:44:36 -05001634 case 0: (a = 0.0); break;
1635 case 1: (a = 1.0); continue;
1636 case 2: (a = 2.0);
1637 default: discard;
1638 }
1639 )");
Ethan Nicholascfefec02021-02-09 15:22:57 -05001640
Ethan Nicholas8a6537d2021-04-30 12:44:00 -04001641 Statement y = StaticSwitch(b,
1642 Case(0, a = 0, Break()),
1643 Case(1, a = 1, Continue()),
1644 Case(2, a = 2 /*Fallthrough*/),
1645 Default(Discard())
1646 );
1647 EXPECT_EQUAL(y, R"(
1648 @switch (b) {
1649 case 0: (a = 0.0); break;
1650 case 1: (a = 1.0); continue;
1651 case 2: (a = 2.0);
1652 default: discard;
1653 }
1654 )");
1655
John Stiles642cde22021-02-23 14:57:01 -05001656 EXPECT_EQUAL(Switch(b),
1657 "switch (b) {}");
1658
1659 EXPECT_EQUAL(Switch(b, Default(), Case(0), Case(1)),
1660 "switch (b) { default: case 0: case 1: }");
Ethan Nicholascfefec02021-02-09 15:22:57 -05001661
1662 {
John Stilese1d1b082021-02-23 13:44:36 -05001663 ExpectError error(r, "error: duplicate case value '0'\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001664 DSLStatement(Switch(0, Case(0), Case(0))).release();
Ethan Nicholascfefec02021-02-09 15:22:57 -05001665 }
1666
1667 {
John Stilese1d1b082021-02-23 13:44:36 -05001668 ExpectError error(r, "error: duplicate default case\n");
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001669 DSLStatement(Switch(0, Default(a = 0), Default(a = 1))).release();
John Stilese1d1b082021-02-23 13:44:36 -05001670 }
1671
1672 {
Ethan Nicholascfefec02021-02-09 15:22:57 -05001673 ExpectError error(r, "error: case value must be a constant integer\n");
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001674 Var b(kInt_Type);
Ethan Nicholas34c7e112021-02-25 20:50:32 -05001675 DSLStatement(Switch(0, Case(b))).release();
Ethan Nicholascfefec02021-02-09 15:22:57 -05001676 }
Ethan Nicholascfefec02021-02-09 15:22:57 -05001677}
1678
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001679DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSwizzle, r, ctxInfo) {
Ethan Nicholas55a63af2021-05-18 10:12:58 -04001680 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001681 Var a(kFloat4_Type, "a");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001682
John Stilesf04e09c2021-03-05 13:13:14 -05001683 EXPECT_EQUAL(a.x(),
1684 "a.x");
1685 EXPECT_EQUAL(a.y(),
1686 "a.y");
1687 EXPECT_EQUAL(a.z(),
1688 "a.z");
1689 EXPECT_EQUAL(a.w(),
1690 "a.w");
1691 EXPECT_EQUAL(a.r(),
1692 "a.x");
1693 EXPECT_EQUAL(a.g(),
1694 "a.y");
1695 EXPECT_EQUAL(a.b(),
1696 "a.z");
1697 EXPECT_EQUAL(a.a(),
1698 "a.w");
1699 EXPECT_EQUAL(Swizzle(a, R),
1700 "a.x");
1701 EXPECT_EQUAL(Swizzle(a, ZERO, G),
1702 "float2(0.0, a.y)");
1703 EXPECT_EQUAL(Swizzle(a, B, G, G),
1704 "a.zyy");
1705 EXPECT_EQUAL(Swizzle(a, R, G, B, ONE),
1706 "float4(a.xyz, 1.0)");
1707 EXPECT_EQUAL(Swizzle(a, B, G, R, ONE).r(),
1708 "a.z");
Ethan Nicholas68c77d42021-01-26 14:31:29 -05001709}
1710
John Stiles08771b02021-04-26 09:35:10 -04001711
1712DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLVarSwap, r, ctxInfo) {
Ethan Nicholas55a63af2021-05-18 10:12:58 -04001713 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), no_mark_vars_declared());
John Stiles08771b02021-04-26 09:35:10 -04001714
1715 // We should be able to convert `a` into a proper var by swapping it, even from within a scope.
1716 Var a;
1717 if (true)
1718 {
1719 Var(kInt_Type, "a").swap(a);
1720 }
1721
1722 EXPECT_EQUAL(Statement(Block(Declare(a), a = 123)),
1723 "{ int a; (a = 123); }");
1724}
1725
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001726DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLWhile, r, ctxInfo) {
1727 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1728 Statement x = While(true, Block());
John Stilesb4d7b582021-02-19 09:56:31 -05001729 EXPECT_EQUAL(x, "for (; true;) {}");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001730
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001731 Var a(kFloat_Type, "a"), b(kFloat_Type, "b");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001732 Statement y = While(a != b, Block(a++, --b));
John Stilesb4d7b582021-02-19 09:56:31 -05001733 EXPECT_EQUAL(y, "for (; (a != b);) { a++; --b; }");
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001734
1735 {
1736 ExpectError error(r, "error: expected 'bool', but found 'int'\n");
Ethan Nicholas549c6b82021-06-25 12:31:44 -04001737 While(7, Block()).release();
Ethan Nicholasd6b6f3e2021-01-22 15:18:25 -05001738 }
1739}
Ethan Nicholas04be3392021-01-26 10:07:01 -05001740
1741DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLIndex, r, ctxInfo) {
1742 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001743 Var a(Array(kInt_Type, 5), "a"), b(kInt_Type, "b");
John Stilesb4d7b582021-02-19 09:56:31 -05001744
1745 EXPECT_EQUAL(a[0], "a[0]");
1746 EXPECT_EQUAL(a[b], "a[b]");
Ethan Nicholas04be3392021-01-26 10:07:01 -05001747
1748 {
1749 ExpectError error(r, "error: expected 'int', but found 'bool'\n");
Ethan Nicholas549c6b82021-06-25 12:31:44 -04001750 a[true].release();
Ethan Nicholas04be3392021-01-26 10:07:01 -05001751 }
1752
1753 {
1754 ExpectError error(r, "error: expected array, but found 'int'\n");
Ethan Nicholas549c6b82021-06-25 12:31:44 -04001755 b[0].release();
Ethan Nicholas04be3392021-01-26 10:07:01 -05001756 }
1757
1758 {
1759 ExpectError error(r, "error: index -1 out of range for 'int[5]'\n");
Ethan Nicholas549c6b82021-06-25 12:31:44 -04001760 a[-1].release();
Ethan Nicholas04be3392021-01-26 10:07:01 -05001761 }
1762}
Ethan Nicholas30e93d52021-01-26 12:00:25 -05001763
1764DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLBuiltins, r, ctxInfo) {
1765 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1766 // There is a Fract type on Mac which can conflict with our Fract builtin
1767 using SkSL::dsl::Fract;
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001768 Var a(kHalf4_Type, "a"), b(kHalf4_Type, "b"), c(kHalf4_Type, "c");
1769 Var h3(kHalf3_Type, "h3");
1770 Var b4(kBool4_Type, "b4");
John Stilesb4d7b582021-02-19 09:56:31 -05001771 EXPECT_EQUAL(Abs(a), "abs(a)");
1772 EXPECT_EQUAL(All(b4), "all(b4)");
1773 EXPECT_EQUAL(Any(b4), "any(b4)");
John Stilese3fa7452021-04-26 09:36:07 -04001774 EXPECT_EQUAL(Atan(a), "atan(a)");
1775 EXPECT_EQUAL(Atan(a, b), "atan(a, b)");
John Stilesb4d7b582021-02-19 09:56:31 -05001776 EXPECT_EQUAL(Ceil(a), "ceil(a)");
1777 EXPECT_EQUAL(Clamp(a, 0, 1), "clamp(a, 0.0, 1.0)");
1778 EXPECT_EQUAL(Cos(a), "cos(a)");
1779 EXPECT_EQUAL(Cross(h3, h3), "cross(h3, h3)");
1780 EXPECT_EQUAL(Degrees(a), "degrees(a)");
1781 EXPECT_EQUAL(Distance(a, b), "distance(a, b)");
1782 EXPECT_EQUAL(Dot(a, b), "dot(a, b)");
1783 EXPECT_EQUAL(Equal(a, b), "equal(a, b)");
1784 EXPECT_EQUAL(Exp(a), "exp(a)");
1785 EXPECT_EQUAL(Exp2(a), "exp2(a)");
1786 EXPECT_EQUAL(Faceforward(a, b, c), "faceforward(a, b, c)");
1787 EXPECT_EQUAL(Floor(a), "floor(a)");
1788 EXPECT_EQUAL(Fract(a), "fract(a)");
1789 EXPECT_EQUAL(GreaterThan(a, b), "greaterThan(a, b)");
1790 EXPECT_EQUAL(GreaterThanEqual(a, b), "greaterThanEqual(a, b)");
1791 EXPECT_EQUAL(Inversesqrt(a), "inversesqrt(a)");
1792 EXPECT_EQUAL(LessThan(a, b), "lessThan(a, b)");
1793 EXPECT_EQUAL(LessThanEqual(a, b), "lessThanEqual(a, b)");
1794 EXPECT_EQUAL(Length(a), "length(a)");
1795 EXPECT_EQUAL(Log(a), "log(a)");
1796 EXPECT_EQUAL(Log2(a), "log2(a)");
1797 EXPECT_EQUAL(Max(a, b), "max(a, b)");
1798 EXPECT_EQUAL(Min(a, b), "min(a, b)");
1799 EXPECT_EQUAL(Mix(a, b, c), "mix(a, b, c)");
1800 EXPECT_EQUAL(Mod(a, b), "mod(a, b)");
1801 EXPECT_EQUAL(Normalize(a), "normalize(a)");
1802 EXPECT_EQUAL(NotEqual(a, b), "notEqual(a, b)");
1803 EXPECT_EQUAL(Pow(a, b), "pow(a, b)");
1804 EXPECT_EQUAL(Radians(a), "radians(a)");
1805 EXPECT_EQUAL(Reflect(a, b), "reflect(a, b)");
1806 EXPECT_EQUAL(Refract(a, b, 1), "refract(a, b, 1.0)");
1807 EXPECT_EQUAL(Saturate(a), "saturate(a)");
1808 EXPECT_EQUAL(Sign(a), "sign(a)");
1809 EXPECT_EQUAL(Sin(a), "sin(a)");
1810 EXPECT_EQUAL(Smoothstep(a, b, c), "smoothstep(a, b, c)");
1811 EXPECT_EQUAL(Sqrt(a), "sqrt(a)");
1812 EXPECT_EQUAL(Step(a, b), "step(a, b)");
1813 EXPECT_EQUAL(Tan(a), "tan(a)");
1814 EXPECT_EQUAL(Unpremul(a), "unpremul(a)");
Ethan Nicholas30e93d52021-01-26 12:00:25 -05001815
1816 // these calls all go through the normal channels, so it ought to be sufficient to prove that
1817 // one of them reports errors correctly
1818 {
1819 ExpectError error(r, "error: no match for ceil(bool)\n");
1820 Ceil(a == b).release();
1821 }
1822}
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001823
1824DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLModifiers, r, ctxInfo) {
Ethan Nicholas55a63af2021-05-18 10:12:58 -04001825 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), no_mark_vars_declared());
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001826
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001827 Var v1(kConst_Modifier, kInt_Type, "v1", 0);
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001828 Statement d1 = Declare(v1);
Ethan Nicholasbd974002021-02-22 16:20:06 -05001829 EXPECT_EQUAL(d1, "const int v1 = 0;");
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001830
1831 // Most modifiers require an appropriate context to be legal. We can't yet give them that
1832 // context, so we can't as yet Declare() variables with these modifiers.
1833 // TODO: better tests when able
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001834 Var v2(kIn_Modifier, kInt_Type, "v2");
Ethan Nicholasb4f8b7a2021-06-23 10:27:09 -04001835 REPORTER_ASSERT(r, v2.modifiers().flags() == SkSL::Modifiers::kIn_Flag);
Ethan Nicholas961d9442021-03-16 16:37:29 -04001836 DSLWriter::MarkDeclared(v2);
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001837
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001838 Var v3(kOut_Modifier, kInt_Type, "v3");
Ethan Nicholasb4f8b7a2021-06-23 10:27:09 -04001839 REPORTER_ASSERT(r, v3.modifiers().flags() == SkSL::Modifiers::kOut_Flag);
Ethan Nicholas961d9442021-03-16 16:37:29 -04001840 DSLWriter::MarkDeclared(v3);
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001841
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001842 Var v4(kFlat_Modifier, kInt_Type, "v4");
Ethan Nicholasb4f8b7a2021-06-23 10:27:09 -04001843 REPORTER_ASSERT(r, v4.modifiers().flags() == SkSL::Modifiers::kFlat_Flag);
Ethan Nicholas961d9442021-03-16 16:37:29 -04001844 DSLWriter::MarkDeclared(v4);
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001845
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001846 Var v5(kNoPerspective_Modifier, kInt_Type, "v5");
Ethan Nicholasb4f8b7a2021-06-23 10:27:09 -04001847 REPORTER_ASSERT(r, v5.modifiers().flags() == SkSL::Modifiers::kNoPerspective_Flag);
Ethan Nicholas961d9442021-03-16 16:37:29 -04001848 DSLWriter::MarkDeclared(v5);
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001849
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001850 Var v6(kIn_Modifier | kOut_Modifier, kInt_Type, "v6");
Ethan Nicholasb4f8b7a2021-06-23 10:27:09 -04001851 REPORTER_ASSERT(r, v6.modifiers().flags() == (SkSL::Modifiers::kIn_Flag |
1852 SkSL::Modifiers::kOut_Flag));
Ethan Nicholas961d9442021-03-16 16:37:29 -04001853 DSLWriter::MarkDeclared(v6);
Ethan Nicholas11a15b12021-02-11 15:56:27 -05001854
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001855 Var v7(kInOut_Modifier, kInt_Type, "v7");
Ethan Nicholasb4f8b7a2021-06-23 10:27:09 -04001856 REPORTER_ASSERT(r, v7.modifiers().flags() == (SkSL::Modifiers::kIn_Flag |
1857 SkSL::Modifiers::kOut_Flag));
Ethan Nicholas961d9442021-03-16 16:37:29 -04001858 DSLWriter::MarkDeclared(v7);
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001859
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001860 Var v8(kUniform_Modifier, kInt_Type, "v8");
Ethan Nicholasb4f8b7a2021-06-23 10:27:09 -04001861 REPORTER_ASSERT(r, v8.modifiers().flags() == SkSL::Modifiers::kUniform_Flag);
Ethan Nicholase9c2c5a2021-04-30 13:14:24 -04001862 DSLWriter::MarkDeclared(v8);
Ethan Nicholas961d9442021-03-16 16:37:29 -04001863 // Uniforms do not need to be explicitly declared
Ethan Nicholasd6b26e52021-01-27 07:53:46 -05001864}
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001865
Ethan Nicholasb22fcaf2021-05-10 16:17:22 -04001866DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLLayout, r, ctxInfo) {
Ethan Nicholas55a63af2021-05-18 10:12:58 -04001867 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), no_mark_vars_declared());
Ethan Nicholasb22fcaf2021-05-10 16:17:22 -04001868 Var v1(DSLModifiers(DSLLayout().location(1).set(2).binding(3).offset(4).index(5).builtin(6)
1869 .inputAttachmentIndex(7),
1870 kConst_Modifier), kInt_Type, "v1", 0);
1871 EXPECT_EQUAL(Declare(v1), "layout (location = 1, offset = 4, binding = 3, index = 5, set = 2, "
1872 "builtin = 6, input_attachment_index = 7) const int v1 = 0;");
1873
1874 Var v2(DSLLayout().originUpperLeft(), kFloat2_Type, "v2");
1875 EXPECT_EQUAL(Declare(v2), "layout (origin_upper_left) float2 v2;");
1876
Ethan Nicholasb22fcaf2021-05-10 16:17:22 -04001877 Var v4(DSLLayout().pushConstant(), kBool_Type, "v4");
1878 EXPECT_EQUAL(Declare(v4), "layout (push_constant) bool v4;");
1879
1880 Var v5(DSLLayout().blendSupportAllEquations(), kHalf4_Type, "v5");
1881 EXPECT_EQUAL(Declare(v5), "layout (blend_support_all_equations) half4 v5;");
1882
Ethan Nicholasb4f8b7a2021-06-23 10:27:09 -04001883 {
1884 ExpectError error(r, "error: 'srgb_unpremul' is only permitted in runtime effects\n");
Ethan Nicholasa2d22b22021-07-15 10:35:54 -04001885 DSLGlobalVar v(DSLModifiers(DSLLayout().srgbUnpremul(), kUniform_Modifier), kHalf4_Type,
1886 "v");
1887 Declare(v);
Ethan Nicholasb4f8b7a2021-06-23 10:27:09 -04001888 }
Ethan Nicholasb22fcaf2021-05-10 16:17:22 -04001889
1890 {
1891 ExpectError error(r, "error: layout qualifier 'location' appears more than once\n");
1892 DSLLayout().location(1).location(2);
1893 }
1894
1895 {
1896 ExpectError error(r, "error: layout qualifier 'set' appears more than once\n");
1897 DSLLayout().set(1).set(2);
1898 }
1899
1900 {
1901 ExpectError error(r, "error: layout qualifier 'binding' appears more than once\n");
1902 DSLLayout().binding(1).binding(2);
1903 }
1904
1905 {
1906 ExpectError error(r, "error: layout qualifier 'offset' appears more than once\n");
1907 DSLLayout().offset(1).offset(2);
1908 }
1909
1910 {
1911 ExpectError error(r, "error: layout qualifier 'index' appears more than once\n");
1912 DSLLayout().index(1).index(2);
1913 }
1914
1915 {
1916 ExpectError error(r, "error: layout qualifier 'builtin' appears more than once\n");
1917 DSLLayout().builtin(1).builtin(2);
1918 }
1919
1920 {
1921 ExpectError error(r, "error: layout qualifier 'input_attachment_index' appears more than "
1922 "once\n");
1923 DSLLayout().inputAttachmentIndex(1).inputAttachmentIndex(2);
1924 }
1925
1926 {
1927 ExpectError error(r, "error: layout qualifier 'origin_upper_left' appears more than "
1928 "once\n");
1929 DSLLayout().originUpperLeft().originUpperLeft();
1930 }
1931
1932 {
Ethan Nicholasb22fcaf2021-05-10 16:17:22 -04001933 ExpectError error(r, "error: layout qualifier 'push_constant' appears more than once\n");
1934 DSLLayout().pushConstant().pushConstant();
1935 }
1936
1937 {
1938 ExpectError error(r, "error: layout qualifier 'blend_support_all_equations' appears more "
1939 "than once\n");
1940 DSLLayout().blendSupportAllEquations().blendSupportAllEquations();
1941 }
1942
1943 {
1944 ExpectError error(r, "error: layout qualifier 'srgb_unpremul' appears more than once\n");
1945 DSLLayout().srgbUnpremul().srgbUnpremul();
1946 }
1947}
1948
Ethan Nicholas624a5292021-04-16 14:54:43 -04001949DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLSampleShader, r, ctxInfo) {
Ethan Nicholas55a63af2021-05-18 10:12:58 -04001950 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), default_settings(),
Brian Osman552fcb92021-04-28 17:41:57 -04001951 SkSL::ProgramKind::kRuntimeShader);
Ethan Nicholasa2d22b22021-07-15 10:35:54 -04001952 DSLGlobalVar shader(kUniform_Modifier, kShader_Type, "shader");
Ethan Nicholas624a5292021-04-16 14:54:43 -04001953 EXPECT_EQUAL(Sample(shader, Float2(0, 0)), "sample(shader, float2(0.0, 0.0))");
Ethan Nicholas624a5292021-04-16 14:54:43 -04001954
1955 {
Brian Osmanc9125aa2021-04-21 09:57:19 -04001956 ExpectError error(r, "error: no match for sample(shader, half4)\n");
Ethan Nicholas624a5292021-04-16 14:54:43 -04001957 Sample(shader, Half4(1)).release();
1958 }
1959}
1960
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001961DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLStruct, r, ctxInfo) {
Ethan Nicholas55a63af2021-05-18 10:12:58 -04001962 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), no_mark_vars_declared());
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001963
1964 DSLType simpleStruct = Struct("SimpleStruct",
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001965 Field(kFloat_Type, "x"),
1966 Field(kBool_Type, "b"),
1967 Field(Array(kFloat_Type, 3), "a")
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001968 );
1969 DSLVar result(simpleStruct, "result");
1970 DSLFunction(simpleStruct, "returnStruct").define(
1971 Declare(result),
1972 result.field("x") = 123,
1973 result.field("b") = result.field("x") > 0,
1974 result.field("a")[0] = result.field("x"),
1975 Return(result)
1976 );
1977 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 2);
John Stilesb4d7b582021-02-19 09:56:31 -05001978 EXPECT_EQUAL(*DSLWriter::ProgramElements()[0],
1979 "struct SimpleStruct { float x; bool b; float[3] a; };");
1980 EXPECT_EQUAL(*DSLWriter::ProgramElements()[1],
1981 "SimpleStruct returnStruct() { SimpleStruct result; (result.x = 123.0);"
1982 "(result.b = (result.x > 0.0)); (result.a[0] = result.x); return result; }");
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001983
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001984 Struct("NestedStruct",
Ethan Nicholasb14e6b92021-04-08 16:56:05 -04001985 Field(kInt_Type, "x"),
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001986 Field(simpleStruct, "simple")
1987 );
Ethan Nicholasfe5d6922021-03-05 14:23:48 -05001988 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 3);
1989 EXPECT_EQUAL(*DSLWriter::ProgramElements()[2],
John Stilesb4d7b582021-02-19 09:56:31 -05001990 "struct NestedStruct { int x; SimpleStruct simple; };");
Ethan Nicholasbf79dff2021-02-11 15:18:31 -05001991}
Ethan Nicholasa1a0b922021-05-04 12:22:02 -04001992
1993DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLWrapper, r, ctxInfo) {
1994 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu());
1995 std::vector<Wrapper<DSLExpression>> exprs;
1996 exprs.push_back(DSLExpression(1));
1997 exprs.emplace_back(2.0);
1998 EXPECT_EQUAL(std::move(*exprs[0]), "1");
1999 EXPECT_EQUAL(std::move(*exprs[1]), "2.0");
2000
2001 std::vector<Wrapper<DSLVar>> vars;
2002 vars.emplace_back(DSLVar(kInt_Type, "x"));
Ethan Nicholasb4f8b7a2021-06-23 10:27:09 -04002003 REPORTER_ASSERT(r, DSLWriter::Var(*vars[0])->name() == "x");
Ethan Nicholasa1a0b922021-05-04 12:22:02 -04002004}
Ethan Nicholasebc9fad2021-07-09 15:35:23 -04002005
2006DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLRTAdjust, r, ctxInfo) {
2007 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), no_mark_vars_declared(),
2008 SkSL::ProgramKind::kVertex);
Ethan Nicholasa2d22b22021-07-15 10:35:54 -04002009 DSLGlobalVar rtAdjust(kUniform_Modifier, kFloat4_Type, "sk_RTAdjust");
2010 Declare(rtAdjust);
Ethan Nicholasebc9fad2021-07-09 15:35:23 -04002011 DSLFunction(kVoid_Type, "main").define(
2012 sk_Position() = Half4(0)
2013 );
2014 REPORTER_ASSERT(r, DSLWriter::ProgramElements().size() == 2);
2015 EXPECT_EQUAL(*DSLWriter::ProgramElements()[1],
2016 "void main() {"
2017 "(sk_PerVertex.sk_Position = float4(0.0));"
2018 "(sk_PerVertex.sk_Position = float4(((sk_PerVertex.sk_Position.xy * sk_RTAdjust.xz) + "
2019 "(sk_PerVertex.sk_Position.ww * sk_RTAdjust.yw)), 0.0, sk_PerVertex.sk_Position.w));"
2020 "}");
2021}
Ethan Nicholas292a09d2021-07-14 09:52:16 -04002022
2023DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLInlining, r, ctxInfo) {
2024 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), no_mark_vars_declared());
Ethan Nicholasa2d22b22021-07-15 10:35:54 -04002025 DSLParameter x(kFloat_Type, "x");
Ethan Nicholas292a09d2021-07-14 09:52:16 -04002026 DSLFunction sqr(kFloat_Type, "sqr", x);
2027 sqr.define(
2028 Return(x * x)
2029 );
2030 DSLFunction(kVoid_Type, "main").define(
2031 sk_FragColor() = (sqr(2), Half4(sqr(3)))
2032 );
Ethan Nicholasb18c1e22021-07-16 09:53:53 -04002033 const char* source = "source test";
2034 std::unique_ptr<SkSL::Program> program = ReleaseProgram(std::make_unique<SkSL::String>(source));
Ethan Nicholas292a09d2021-07-14 09:52:16 -04002035 EXPECT_EQUAL(*program,
2036 "layout(location = 0, index = 0, builtin = 10001) out half4 sk_FragColor;"
2037 "layout(builtin = 17)in bool sk_Clockwise;"
2038 "void main() {"
2039 "/* inlined: sqr */;"
2040 "/* inlined: sqr */;"
2041 "(sk_FragColor = (4.0 , half4(half(9.0))));"
2042 "}");
Ethan Nicholasb18c1e22021-07-16 09:53:53 -04002043 REPORTER_ASSERT(r, *program->fSource == source);
Ethan Nicholas292a09d2021-07-14 09:52:16 -04002044}
Ethan Nicholas459777a2021-07-16 11:16:27 -04002045
2046DEF_GPUTEST_FOR_MOCK_CONTEXT(DSLReleaseUnused, r, ctxInfo) {
2047 SkSL::ProgramSettings settings = default_settings();
2048 settings.fAssertDSLObjectsReleased = false;
2049 AutoDSLContext context(ctxInfo.directContext()->priv().getGpu(), settings);
2050 If(Sqrt(1) > 0, Discard());
2051 // Ensure that we can safely destroy statements and expressions despite being unused while
2052 // settings.fAssertDSLObjectsReleased is disabled.
2053}