// Copyright 2022 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/tint/transform/while_to_loop.h" #include "src/tint/ast/break_statement.h" #include "src/tint/program_builder.h" TINT_INSTANTIATE_TYPEINFO(tint::transform::WhileToLoop); namespace tint::transform { WhileToLoop::WhileToLoop() = default; WhileToLoop::~WhileToLoop() = default; bool WhileToLoop::ShouldRun(const Program* program, const DataMap&) const { for (auto* node : program->ASTNodes().Objects()) { if (node->Is()) { return true; } } return false; } void WhileToLoop::Run(CloneContext& ctx, const DataMap&, DataMap&) const { ctx.ReplaceAll([&](const ast::WhileStatement* w) -> const ast::Statement* { utils::Vector stmts; auto* cond = w->condition; // !condition auto* not_cond = ctx.dst->create(ast::UnaryOp::kNot, ctx.Clone(cond)); // { break; } auto* break_body = ctx.dst->Block(ctx.dst->create()); // if (!condition) { break; } stmts.Push(ctx.dst->If(not_cond, break_body)); for (auto* stmt : w->body->statements) { stmts.Push(ctx.Clone(stmt)); } const ast::BlockStatement* continuing = nullptr; auto* body = ctx.dst->Block(stmts); auto* loop = ctx.dst->create(body, continuing); return loop; }); ctx.Clone(); } } // namespace tint::transform