2021-07-02 22:17:25 +00:00
|
|
|
// Copyright 2021 The Tint Authors.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
#include "src/transform/loop_to_for_loop.h"
|
|
|
|
|
|
|
|
#include "src/transform/test_helper.h"
|
|
|
|
|
|
|
|
namespace tint {
|
|
|
|
namespace transform {
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
using LoopToForLoopTest = TransformTest;
|
|
|
|
|
|
|
|
TEST_F(LoopToForLoopTest, EmptyModule) {
|
|
|
|
auto* src = "";
|
|
|
|
auto* expect = "";
|
|
|
|
|
|
|
|
auto got = Run<LoopToForLoop>(src);
|
|
|
|
|
|
|
|
EXPECT_EQ(expect, str(got));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(LoopToForLoopTest, IfBreak) {
|
|
|
|
auto* src = R"(
|
|
|
|
fn f() {
|
|
|
|
var i : i32;
|
|
|
|
i = 0;
|
|
|
|
loop {
|
|
|
|
if (i > 15) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ignore(123);
|
|
|
|
|
|
|
|
continuing {
|
|
|
|
i = i + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
|
|
|
|
auto* expect = R"(
|
|
|
|
fn f() {
|
|
|
|
var i : i32;
|
|
|
|
i = 0;
|
|
|
|
for(; !((i > 15)); i = (i + 1)) {
|
|
|
|
ignore(123);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
|
|
|
|
auto got = Run<LoopToForLoop>(src);
|
|
|
|
|
|
|
|
EXPECT_EQ(expect, str(got));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(LoopToForLoopTest, IfElseBreak) {
|
|
|
|
auto* src = R"(
|
|
|
|
fn f() {
|
|
|
|
var i : i32;
|
|
|
|
i = 0;
|
|
|
|
loop {
|
|
|
|
if (i < 15) {
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ignore(123);
|
|
|
|
|
|
|
|
continuing {
|
|
|
|
i = i + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
|
|
|
|
auto* expect = R"(
|
|
|
|
fn f() {
|
|
|
|
var i : i32;
|
|
|
|
i = 0;
|
|
|
|
for(; (i < 15); i = (i + 1)) {
|
|
|
|
ignore(123);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
|
|
|
|
auto got = Run<LoopToForLoop>(src);
|
|
|
|
|
|
|
|
EXPECT_EQ(expect, str(got));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(LoopToForLoopTest, Nested) {
|
|
|
|
auto* src = R"(
|
|
|
|
let N = 16u;
|
|
|
|
|
|
|
|
fn f() {
|
|
|
|
var i : u32 = 0u;
|
|
|
|
loop {
|
|
|
|
if (i >= N) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
{
|
|
|
|
var j : u32 = 0u;
|
|
|
|
loop {
|
|
|
|
if (j >= N) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ignore(i);
|
|
|
|
ignore(j);
|
|
|
|
|
|
|
|
continuing {
|
|
|
|
j = (j + 1u);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
continuing {
|
|
|
|
i = (i + 1u);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
|
|
|
|
auto* expect = R"(
|
|
|
|
let N = 16u;
|
|
|
|
|
|
|
|
fn f() {
|
|
|
|
var i : u32 = 0u;
|
|
|
|
for(; !((i >= N)); i = (i + 1u)) {
|
|
|
|
{
|
|
|
|
var j : u32 = 0u;
|
|
|
|
for(; !((j >= N)); j = (j + 1u)) {
|
|
|
|
ignore(i);
|
|
|
|
ignore(j);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
|
|
|
|
auto got = Run<LoopToForLoop>(src);
|
|
|
|
|
|
|
|
EXPECT_EQ(expect, str(got));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(LoopToForLoopTest, NoTransform_IfMultipleStmts) {
|
|
|
|
auto* src = R"(
|
|
|
|
fn f() {
|
|
|
|
var i : i32;
|
|
|
|
i = 0;
|
|
|
|
loop {
|
|
|
|
if ((i < 15)) {
|
|
|
|
ignore(i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
ignore(123);
|
|
|
|
|
|
|
|
continuing {
|
|
|
|
i = (i + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
|
|
|
|
auto* expect = src;
|
|
|
|
|
|
|
|
auto got = Run<LoopToForLoop>(src);
|
|
|
|
|
|
|
|
EXPECT_EQ(expect, str(got));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(LoopToForLoopTest, NoTransform_IfElseMultipleStmts) {
|
|
|
|
auto* src = R"(
|
|
|
|
fn f() {
|
|
|
|
var i : i32;
|
|
|
|
i = 0;
|
|
|
|
loop {
|
|
|
|
if ((i < 15)) {
|
|
|
|
} else {
|
|
|
|
ignore(i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
ignore(123);
|
|
|
|
|
|
|
|
continuing {
|
|
|
|
i = (i + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
|
|
|
|
auto* expect = src;
|
|
|
|
|
|
|
|
auto got = Run<LoopToForLoop>(src);
|
|
|
|
|
|
|
|
EXPECT_EQ(expect, str(got));
|
|
|
|
}
|
|
|
|
|
2021-07-29 18:05:19 +00:00
|
|
|
TEST_F(LoopToForLoopTest, NoTransform_ContinuingIsCompound) {
|
|
|
|
auto* src = R"(
|
|
|
|
fn f() {
|
|
|
|
var i : i32;
|
|
|
|
i = 0;
|
|
|
|
loop {
|
|
|
|
if ((i < 15)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
ignore(123);
|
|
|
|
|
|
|
|
continuing {
|
|
|
|
if (false) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
|
|
|
|
auto* expect = src;
|
|
|
|
|
|
|
|
auto got = Run<LoopToForLoop>(src);
|
|
|
|
|
|
|
|
EXPECT_EQ(expect, str(got));
|
|
|
|
}
|
|
|
|
|
2021-07-02 22:17:25 +00:00
|
|
|
TEST_F(LoopToForLoopTest, NoTransform_ContinuingMultipleStmts) {
|
|
|
|
auto* src = R"(
|
|
|
|
fn f() {
|
|
|
|
var i : i32;
|
|
|
|
i = 0;
|
|
|
|
loop {
|
|
|
|
if ((i < 15)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
ignore(123);
|
|
|
|
|
|
|
|
continuing {
|
|
|
|
i = (i + 1);
|
|
|
|
ignore(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
|
|
|
|
auto* expect = src;
|
|
|
|
|
|
|
|
auto got = Run<LoopToForLoop>(src);
|
|
|
|
|
|
|
|
EXPECT_EQ(expect, str(got));
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(LoopToForLoopTest, NoTransform_ContinuingUsesVarDeclInLoopBody) {
|
|
|
|
auto* src = R"(
|
|
|
|
fn f() {
|
|
|
|
var i : i32;
|
|
|
|
i = 0;
|
|
|
|
loop {
|
|
|
|
if ((i < 15)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
var j : i32;
|
|
|
|
|
|
|
|
continuing {
|
|
|
|
i = (i + j);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)";
|
|
|
|
|
|
|
|
auto* expect = src;
|
|
|
|
|
|
|
|
auto got = Run<LoopToForLoop>(src);
|
|
|
|
|
|
|
|
EXPECT_EQ(expect, str(got));
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
} // namespace transform
|
|
|
|
} // namespace tint
|