writer/wgsl: Fix printing of for-loops

Fix various issue with formatting for loop. Add tests.

Bug: tint:952
Change-Id: I704341a15f0050ebf82df219d0c7d068a3a63c26
Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/58064
Kokoro: Kokoro <noreply+kokoro@google.com>
Reviewed-by: James Price <jrprice@google.com>
This commit is contained in:
Ben Clayton 2021-07-15 22:20:29 +00:00
parent 8e38fad091
commit 0bff3fb3b7
14 changed files with 211 additions and 15 deletions

View File

@ -181,7 +181,7 @@ TEST_F(MslGeneratorImplTest, Emit_ForLoopWithSimpleInit) {
TEST_F(MslGeneratorImplTest, Emit_ForLoopWithMultiStmtInit) { TEST_F(MslGeneratorImplTest, Emit_ForLoopWithMultiStmtInit) {
// var<workgroup> a : atomic<i32>; // var<workgroup> a : atomic<i32>;
// for(var b = atomicCompareExchangeWeak(&a, 1, 2); ; ) { // for({ignore(1); ignore(2);}; ; ) {
// return; // return;
// } // }
Global("a", ty.atomic<i32>(), ast::StorageClass::kWorkgroup); Global("a", ty.atomic<i32>(), ast::StorageClass::kWorkgroup);

View File

@ -108,11 +108,11 @@ void TextGenerator::TextBuffer::Insert(const TextBuffer& tb,
} }
} }
std::string TextGenerator::TextBuffer::String() const { std::string TextGenerator::TextBuffer::String(uint32_t indent /* = 0 */) const {
std::stringstream ss; std::stringstream ss;
for (auto& line : lines) { for (auto& line : lines) {
if (!line.content.empty()) { if (!line.content.empty()) {
for (uint32_t i = 0; i < line.indent; i++) { for (uint32_t i = 0; i < indent + line.indent; i++) {
ss << " "; ss << " ";
} }
ss << line.content; ss << line.content;

View File

@ -106,7 +106,8 @@ class TextGenerator {
void Insert(const TextBuffer& tb, size_t before, uint32_t indent); void Insert(const TextBuffer& tb, size_t before, uint32_t indent);
/// @returns the buffer's content as a single string /// @returns the buffer's content as a single string
std::string String() const; /// @param indent additional indentation to apply to each line
std::string String(uint32_t indent = 0) const;
/// The current indentation of the TextBuffer. Lines appended to the /// The current indentation of the TextBuffer. Lines appended to the
/// TextBuffer will use this indentation. /// TextBuffer will use this indentation.

View File

@ -1034,17 +1034,20 @@ bool GeneratorImpl::EmitForLoop(ast::ForLoopStatement* stmt) {
ScopedParen sp(out); ScopedParen sp(out);
switch (init_buf.lines.size()) { switch (init_buf.lines.size()) {
case 0: // No initializer case 0: // No initializer
out << ";";
break; break;
case 1: // Single line initializer statement case 1: // Single line initializer statement
out << init_buf.lines[0].content; out << TrimSuffix(init_buf.lines[0].content, ";");
break; break;
default: // Block initializer statement default: // Block initializer statement
current_buffer_->Append(init_buf); for (size_t i = 1; i < init_buf.lines.size(); i++) {
// Indent all by the first line
init_buf.lines[i].indent += current_buffer_->current_indent;
}
out << TrimSuffix(init_buf.String(), "\n");
break; break;
} }
out << " "; out << "; ";
if (auto* cond = stmt->condition()) { if (auto* cond = stmt->condition()) {
if (!EmitExpression(out, cond)) { if (!EmitExpression(out, cond)) {
@ -1056,13 +1059,16 @@ bool GeneratorImpl::EmitForLoop(ast::ForLoopStatement* stmt) {
switch (cont_buf.lines.size()) { switch (cont_buf.lines.size()) {
case 0: // No continuing case 0: // No continuing
out << ";";
break; break;
case 1: // Single line continuing statement case 1: // Single line continuing statement
out << TrimSuffix(cont_buf.lines[0].content, ";"); out << TrimSuffix(cont_buf.lines[0].content, ";");
break; break;
default: // Block continuing statement default: // Block continuing statement
current_buffer_->Append(cont_buf); for (size_t i = 1; i < cont_buf.lines.size(); i++) {
// Indent all by the first line
cont_buf.lines[i].indent += current_buffer_->current_indent;
}
out << TrimSuffix(cont_buf.String(), "\n");
break; break;
} }
} }

View File

@ -61,6 +61,142 @@ TEST_F(WgslGeneratorImplTest, Emit_LoopWithContinuing) {
)"); )");
} }
TEST_F(WgslGeneratorImplTest, Emit_ForLoopWithMultiStmtInit) {
// var<workgroup> a : atomic<i32>;
// for({ignore(1); ignore(2);}; ; ) {
// return;
// }
Global("a", ty.atomic<i32>(), ast::StorageClass::kWorkgroup);
auto* multi_stmt = Block(Ignore(1), Ignore(2));
auto* f = For(multi_stmt, nullptr, nullptr, Block(Return()));
WrapInFunction(f);
GeneratorImpl& gen = Build();
gen.increment_indent();
ASSERT_TRUE(gen.EmitStatement(f)) << gen.error();
EXPECT_EQ(gen.result(), R"( for({
ignore(1);
ignore(2);
}; ; ) {
return;
}
)");
}
TEST_F(WgslGeneratorImplTest, Emit_ForLoopWithSimpleCond) {
// for(; true; ) {
// return;
// }
auto* f = For(nullptr, true, nullptr, Block(Return()));
WrapInFunction(f);
GeneratorImpl& gen = Build();
gen.increment_indent();
ASSERT_TRUE(gen.EmitStatement(f)) << gen.error();
EXPECT_EQ(gen.result(), R"( for(; true; ) {
return;
}
)");
}
TEST_F(WgslGeneratorImplTest, Emit_ForLoopWithSimpleCont) {
// for(; ; i = i + 1) {
// return;
// }
auto* v = Decl(Var("i", ty.i32()));
auto* f = For(nullptr, nullptr, Assign("i", Add("i", 1)), Block(Return()));
WrapInFunction(v, f);
GeneratorImpl& gen = Build();
gen.increment_indent();
ASSERT_TRUE(gen.EmitStatement(f)) << gen.error();
EXPECT_EQ(gen.result(), R"( for(; ; i = (i + 1)) {
return;
}
)");
}
TEST_F(WgslGeneratorImplTest, Emit_ForLoopWithMultiStmtCont) {
// var<workgroup> a : atomic<i32>;
// for(; ; { ignore(1); ignore(2); }) {
// return;
// }
Global("a", ty.atomic<i32>(), ast::StorageClass::kWorkgroup);
auto* multi_stmt = Block(Ignore(1), Ignore(2));
auto* f = For(nullptr, nullptr, multi_stmt, Block(Return()));
WrapInFunction(f);
GeneratorImpl& gen = Build();
gen.increment_indent();
ASSERT_TRUE(gen.EmitStatement(f)) << gen.error();
EXPECT_EQ(gen.result(), R"( for(; ; {
ignore(1);
ignore(2);
}) {
return;
}
)");
}
TEST_F(WgslGeneratorImplTest, Emit_ForLoopWithSimpleInitCondCont) {
// for(var i : i32; true; i = i + 1) {
// return;
// }
auto* f = For(Decl(Var("i", ty.i32())), true, Assign("i", Add("i", 1)),
Block(Return()));
WrapInFunction(f);
GeneratorImpl& gen = Build();
gen.increment_indent();
ASSERT_TRUE(gen.EmitStatement(f)) << gen.error();
EXPECT_EQ(gen.result(), R"( for(var i : i32; true; i = (i + 1)) {
return;
}
)");
}
TEST_F(WgslGeneratorImplTest, Emit_ForLoopWithMultiStmtInitCondCont) {
// var<workgroup> a : atomic<i32>;
// for({ ignore(1); ignore(2); }; true; { ignore(3); ignore(4); }) {
// return;
// }
Global("a", ty.atomic<i32>(), ast::StorageClass::kWorkgroup);
auto* multi_stmt_a = Block(Ignore(1), Ignore(2));
auto* multi_stmt_b = Block(Ignore(3), Ignore(4));
auto* f = For(multi_stmt_a, Expr(true), multi_stmt_b, Block(Return()));
WrapInFunction(f);
GeneratorImpl& gen = Build();
gen.increment_indent();
ASSERT_TRUE(gen.EmitStatement(f)) << gen.error();
EXPECT_EQ(gen.result(), R"( for({
ignore(1);
ignore(2);
}; true; {
ignore(3);
ignore(4);
}) {
return;
}
)");
}
} // namespace } // namespace
} // namespace wgsl } // namespace wgsl
} // namespace writer } // namespace writer

View File

@ -1,5 +1,5 @@
fn f() { fn f() {
var i : i32; var i : i32;
for(let p = &(i); ; ;) { for(let p = &(i); ; ) {
} }
} }

View File

@ -1,5 +1,5 @@
fn f() { fn f() {
var i : i32; var i : i32;
for(; (i < 4); ;) { for(; (i < 4); ) {
} }
} }

View File

@ -1,4 +1,4 @@
fn f() { fn f() {
for(; ; ;) { for(; ; ) {
} }
} }

View File

@ -1,4 +1,4 @@
fn f() { fn f() {
for(var i : i32 = 0; ; ;) { for(var i : i32 = 0; ; ) {
} }
} }

View File

@ -1,5 +1,5 @@
fn f() { fn f() {
for(var must_not_collide : i32 = 0; ; ;) { for(var must_not_collide : i32 = 0; ; ) {
} }
var must_not_collide : i32; var must_not_collide : i32;
} }

View File

@ -0,0 +1,11 @@
[numthreads(1, 1, 1)]
void unused_entry_point() {
return;
}
void f() {
{
for(; ; ) {
}
}
}

View File

@ -0,0 +1,8 @@
#include <metal_stdlib>
using namespace metal;
void f() {
for(; ; ) {
}
}

View File

@ -0,0 +1,30 @@
; SPIR-V
; Version: 1.3
; Generator: Google Tint Compiler; 0
; Bound: 11
; Schema: 0
OpCapability Shader
OpMemoryModel Logical GLSL450
OpEntryPoint GLCompute %unused_entry_point "unused_entry_point"
OpExecutionMode %unused_entry_point LocalSize 1 1 1
OpName %unused_entry_point "unused_entry_point"
OpName %f "f"
%void = OpTypeVoid
%1 = OpTypeFunction %void
%unused_entry_point = OpFunction %void None %1
%4 = OpLabel
OpReturn
OpFunctionEnd
%f = OpFunction %void None %1
%6 = OpLabel
OpBranch %7
%7 = OpLabel
OpLoopMerge %8 %9 None
OpBranch %10
%10 = OpLabel
OpBranch %9
%9 = OpLabel
OpBranch %7
%8 = OpLabel
OpReturn
OpFunctionEnd

View File

@ -0,0 +1,4 @@
fn f() {
for(; ; ) {
}
}