blob: 97aaa19bb57a02435a99f9238c947a72acc6efef [file] [log] [blame]
dan sinclairc0258db2024-06-18 17:13:49 +00001// Copyright 2024 The Dawn & Tint Authors
2//
3// Redistribution and use in source and binary forms, with or without
4// modification, are permitted provided that the following conditions are met:
5//
6// 1. Redistributions of source code must retain the above copyright notice, this
7// list of conditions and the following disclaimer.
8//
9// 2. Redistributions in binary form must reproduce the above copyright notice,
10// this list of conditions and the following disclaimer in the documentation
11// and/or other materials provided with the distribution.
12//
13// 3. Neither the name of the copyright holder nor the names of its
14// contributors may be used to endorse or promote products derived from
15// this software without specific prior written permission.
16//
17// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
18// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
20// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
21// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
23// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
24// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
25// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
28#include "src/tint/lang/hlsl/writer/helper_test.h"
29
30using namespace tint::core::number_suffixes; // NOLINT
31
32namespace tint::hlsl::writer {
33namespace {
34
35TEST_F(HlslWriterTest, If) {
36 auto* func = b.Function("foo", ty.void_(), core::ir::Function::PipelineStage::kCompute);
37 func->SetWorkgroupSize(1, 1, 1);
38 b.Append(func->Block(), [&] {
39 auto* if_ = b.If(true);
40 b.Append(if_->True(), [&] { b.ExitIf(if_); });
41 b.Return(func);
42 });
43
44 ASSERT_TRUE(Generate()) << err_ << output_.hlsl;
45 EXPECT_EQ(output_.hlsl, R"(
46[numthreads(1, 1, 1)]
47void foo() {
48 if (true) {
49 }
50}
51
52)");
53}
54
55TEST_F(HlslWriterTest, IfWithElseIf) {
56 auto* func = b.Function("foo", ty.void_(), core::ir::Function::PipelineStage::kCompute);
57 func->SetWorkgroupSize(1, 1, 1);
58 b.Append(func->Block(), [&] {
59 auto* if_ = b.If(true);
60 b.Append(if_->True(), [&] { b.ExitIf(if_); });
61 b.Append(if_->False(), [&] {
62 auto* false_ = b.If(false);
63 b.Append(false_->True(), [&] { b.ExitIf(false_); });
64 b.ExitIf(if_);
65 });
66 b.Return(func);
67 });
68
69 ASSERT_TRUE(Generate()) << err_ << output_.hlsl;
70 EXPECT_EQ(output_.hlsl, R"(
71[numthreads(1, 1, 1)]
72void foo() {
73 if (true) {
74 } else {
75 if (false) {
76 }
77 }
78}
79
80)");
81}
82
83TEST_F(HlslWriterTest, IfWithElse) {
84 auto* func = b.Function("foo", ty.void_(), core::ir::Function::PipelineStage::kCompute);
85 func->SetWorkgroupSize(1, 1, 1);
86 b.Append(func->Block(), [&] {
87 auto* if_ = b.If(true);
88 b.Append(if_->True(), [&] { b.ExitIf(if_); });
89 b.Append(if_->False(), [&] { b.Return(func); });
90 b.Return(func);
91 });
92
93 ASSERT_TRUE(Generate()) << err_ << output_.hlsl;
94 EXPECT_EQ(output_.hlsl, R"(
95[numthreads(1, 1, 1)]
96void foo() {
97 if (true) {
98 } else {
99 return;
100 }
101}
102
103)");
104}
105
106TEST_F(HlslWriterTest, IfBothBranchesReturn) {
107 auto* func = b.Function("foo", ty.void_(), core::ir::Function::PipelineStage::kCompute);
108 func->SetWorkgroupSize(1, 1, 1);
109 b.Append(func->Block(), [&] {
110 auto* if_ = b.If(true);
111 b.Append(if_->True(), [&] { b.Return(func); });
112 b.Append(if_->False(), [&] { b.Return(func); });
113 b.Unreachable();
114 });
115
116 ASSERT_TRUE(Generate()) << err_ << output_.hlsl;
117 EXPECT_EQ(output_.hlsl, R"(
118[numthreads(1, 1, 1)]
119void foo() {
120 if (true) {
121 return;
122 } else {
123 return;
124 }
125 /* unreachable */
126}
127
128)");
129}
130
131TEST_F(HlslWriterTest, IfWithSinglePhi) {
132 auto* func = b.Function("foo", ty.void_(), core::ir::Function::PipelineStage::kCompute);
133 func->SetWorkgroupSize(1, 1, 1);
134 b.Append(func->Block(), [&] {
135 auto* i = b.If(true);
136 i->SetResults(b.InstructionResult(ty.i32()));
137 b.Append(i->True(), [&] { //
138 b.ExitIf(i, 10_i);
139 });
140 b.Append(i->False(), [&] { //
141 b.ExitIf(i, 20_i);
142 });
143 b.Return(func);
144 });
145
146 ASSERT_TRUE(Generate()) << err_ << output_.hlsl;
147 EXPECT_EQ(output_.hlsl, R"(
148[numthreads(1, 1, 1)]
149void foo() {
Antonio Maioranod032c622024-09-19 18:20:54 +0000150 int v = int(0);
dan sinclairc0258db2024-06-18 17:13:49 +0000151 if (true) {
Antonio Maioranod032c622024-09-19 18:20:54 +0000152 v = int(10);
dan sinclairc0258db2024-06-18 17:13:49 +0000153 } else {
Antonio Maioranod032c622024-09-19 18:20:54 +0000154 v = int(20);
dan sinclairc0258db2024-06-18 17:13:49 +0000155 }
156}
157
158)");
159}
160
161TEST_F(HlslWriterTest, IfWithMultiPhi) {
162 auto* func = b.Function("foo", ty.void_(), core::ir::Function::PipelineStage::kCompute);
163 func->SetWorkgroupSize(1, 1, 1);
164 b.Append(func->Block(), [&] {
165 auto* i = b.If(true);
166 i->SetResults(b.InstructionResult(ty.i32()), b.InstructionResult(ty.bool_()));
167 b.Append(i->True(), [&] { //
168 b.ExitIf(i, 10_i, true);
169 });
170 b.Append(i->False(), [&] { //
171 b.ExitIf(i, 20_i, false);
172 });
173 b.Return(func);
174 });
175
176 ASSERT_TRUE(Generate()) << err_ << output_.hlsl;
177 EXPECT_EQ(output_.hlsl, R"(
178[numthreads(1, 1, 1)]
179void foo() {
Antonio Maioranod032c622024-09-19 18:20:54 +0000180 int v = int(0);
dan sinclairc0258db2024-06-18 17:13:49 +0000181 bool v_1 = false;
182 if (true) {
Antonio Maioranod032c622024-09-19 18:20:54 +0000183 v = int(10);
dan sinclairc0258db2024-06-18 17:13:49 +0000184 v_1 = true;
185 } else {
Antonio Maioranod032c622024-09-19 18:20:54 +0000186 v = int(20);
dan sinclairc0258db2024-06-18 17:13:49 +0000187 v_1 = false;
188 }
189}
190
191)");
192}
193
194TEST_F(HlslWriterTest, IfWithMultiPhiReturn1) {
dan sinclaire4321b92024-06-27 15:36:11 +0000195 auto* func = b.Function("foo", ty.i32());
dan sinclairc0258db2024-06-18 17:13:49 +0000196 b.Append(func->Block(), [&] {
197 auto* i = b.If(true);
198 i->SetResults(b.InstructionResult(ty.i32()), b.InstructionResult(ty.bool_()));
199 b.Append(i->True(), [&] { //
200 b.ExitIf(i, 10_i, true);
201 });
202 b.Append(i->False(), [&] { //
203 b.ExitIf(i, 20_i, false);
204 });
205 b.Return(func, i->Result(0));
206 });
207
208 ASSERT_TRUE(Generate()) << err_ << output_.hlsl;
209 EXPECT_EQ(output_.hlsl, R"(
dan sinclairc0258db2024-06-18 17:13:49 +0000210int foo() {
Antonio Maioranod032c622024-09-19 18:20:54 +0000211 int v = int(0);
dan sinclairc0258db2024-06-18 17:13:49 +0000212 bool v_1 = false;
213 if (true) {
Antonio Maioranod032c622024-09-19 18:20:54 +0000214 v = int(10);
dan sinclairc0258db2024-06-18 17:13:49 +0000215 v_1 = true;
216 } else {
Antonio Maioranod032c622024-09-19 18:20:54 +0000217 v = int(20);
dan sinclairc0258db2024-06-18 17:13:49 +0000218 v_1 = false;
219 }
220 return v;
221}
222
dan sinclaire4321b92024-06-27 15:36:11 +0000223[numthreads(1, 1, 1)]
224void unused_entry_point() {
225}
226
dan sinclairc0258db2024-06-18 17:13:49 +0000227)");
228}
229
230TEST_F(HlslWriterTest, IfWithMultiPhiReturn2) {
dan sinclaire4321b92024-06-27 15:36:11 +0000231 auto* func = b.Function("foo", ty.bool_());
dan sinclairc0258db2024-06-18 17:13:49 +0000232 b.Append(func->Block(), [&] {
233 auto* i = b.If(true);
234 i->SetResults(b.InstructionResult(ty.i32()), b.InstructionResult(ty.bool_()));
235 b.Append(i->True(), [&] { //
236 b.ExitIf(i, 10_i, true);
237 });
238 b.Append(i->False(), [&] { //
239 b.ExitIf(i, 20_i, false);
240 });
241 b.Return(func, i->Result(1));
242 });
243
244 ASSERT_TRUE(Generate()) << err_ << output_.hlsl;
245 EXPECT_EQ(output_.hlsl, R"(
dan sinclairc0258db2024-06-18 17:13:49 +0000246bool foo() {
Antonio Maioranod032c622024-09-19 18:20:54 +0000247 int v = int(0);
dan sinclairc0258db2024-06-18 17:13:49 +0000248 bool v_1 = false;
249 if (true) {
Antonio Maioranod032c622024-09-19 18:20:54 +0000250 v = int(10);
dan sinclairc0258db2024-06-18 17:13:49 +0000251 v_1 = true;
252 } else {
Antonio Maioranod032c622024-09-19 18:20:54 +0000253 v = int(20);
dan sinclairc0258db2024-06-18 17:13:49 +0000254 v_1 = false;
255 }
256 return v_1;
257}
258
dan sinclaire4321b92024-06-27 15:36:11 +0000259[numthreads(1, 1, 1)]
260void unused_entry_point() {
261}
262
dan sinclairc0258db2024-06-18 17:13:49 +0000263)");
264}
265
266} // namespace
267} // namespace tint::hlsl::writer