blob: 3988710ed66f4afdd762ac08be4f751d20b2c5c2 [file] [log] [blame]
// 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 <algorithm>
#include "src/ast/block_statement.h"
#include "src/ast/loop_statement.h"
#include "src/ast/statement.h"
#include "src/debug.h"
#include "src/sem/block_statement.h"
#include "src/sem/statement.h"
TINT_INSTANTIATE_TYPEINFO(tint::sem::Statement);
namespace tint {
namespace sem {
Statement::Statement(const ast::Statement* declaration, const Statement* parent)
: declaration_(declaration), parent_(parent) {
#ifndef NDEBUG
if (parent_) {
auto* block = Block();
if (parent_ == block) {
// The parent of this statement is a block. We thus expect the statement
// to be an element of the block. There is one exception: a loop's
// continuing block has the loop's body as its parent, but the continuing
// block is not a statement in the body, so we rule out that case.
auto& stmts = block->Declaration()->statements();
if (std::find(stmts.begin(), stmts.end(), declaration) == stmts.end()) {
bool statement_is_continuing_for_loop = false;
if (parent_->parent_ != nullptr) {
if (auto* loop =
parent_->parent_->Declaration()->As<ast::LoopStatement>()) {
if (loop->has_continuing() && Declaration() == loop->continuing()) {
statement_is_continuing_for_loop = true;
}
}
}
TINT_ASSERT(statement_is_continuing_for_loop);
}
}
}
#endif // NDEBUG
}
const BlockStatement* Statement::Block() const {
auto* stmt = parent_;
while (stmt != nullptr) {
if (auto* block_stmt = stmt->As<BlockStatement>()) {
return block_stmt;
}
stmt = stmt->parent_;
}
return nullptr;
}
const ast::Function* Statement::Function() const {
if (auto* block = Block()) {
if (auto* fbs = block->FindFirstParent<FunctionBlockStatement>()) {
return fbs->Function();
}
}
return nullptr;
}
} // namespace sem
} // namespace tint