2016-11-10 17:11:50 +00:00
|
|
|
// Copyright (c) 2016 Google Inc.
|
|
|
|
//
|
|
|
|
// 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 "basic_block.h"
|
2017-11-14 19:11:50 +00:00
|
|
|
#include "function.h"
|
|
|
|
#include "module.h"
|
2016-11-10 17:11:50 +00:00
|
|
|
|
2017-07-13 00:16:51 +00:00
|
|
|
#include "make_unique.h"
|
|
|
|
|
2016-11-10 17:11:50 +00:00
|
|
|
namespace spvtools {
|
|
|
|
namespace ir {
|
|
|
|
|
2017-10-30 21:42:26 +00:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
const uint32_t kLoopMergeContinueBlockIdInIdx = 1;
|
|
|
|
const uint32_t kLoopMergeMergeBlockIdInIdx = 0;
|
|
|
|
const uint32_t kSelectionMergeMergeBlockIdInIdx = 0;
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
2017-11-14 19:11:50 +00:00
|
|
|
BasicBlock* BasicBlock::Clone(IRContext* context) const {
|
2017-11-27 15:16:41 +00:00
|
|
|
BasicBlock* clone = new BasicBlock(
|
|
|
|
std::unique_ptr<Instruction>(GetLabelInst().Clone(context)));
|
2017-11-14 19:11:50 +00:00
|
|
|
for (const auto& inst : insts_)
|
|
|
|
// Use the incoming context
|
|
|
|
clone->AddInstruction(std::unique_ptr<Instruction>(inst.Clone(context)));
|
|
|
|
return clone;
|
2017-07-13 00:16:51 +00:00
|
|
|
}
|
|
|
|
|
2017-08-31 15:37:17 +00:00
|
|
|
const Instruction* BasicBlock::GetMergeInst() const {
|
|
|
|
const Instruction* result = nullptr;
|
|
|
|
// If it exists, the merge instruction immediately precedes the
|
|
|
|
// terminator.
|
|
|
|
auto iter = ctail();
|
|
|
|
if (iter != cbegin()) {
|
|
|
|
--iter;
|
|
|
|
const auto opcode = iter->opcode();
|
2017-10-30 21:42:26 +00:00
|
|
|
if (opcode == SpvOpLoopMerge || opcode == SpvOpSelectionMerge) {
|
2017-08-31 15:37:17 +00:00
|
|
|
result = &*iter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
Instruction* BasicBlock::GetMergeInst() {
|
|
|
|
Instruction* result = nullptr;
|
|
|
|
// If it exists, the merge instruction immediately precedes the
|
|
|
|
// terminator.
|
|
|
|
auto iter = tail();
|
|
|
|
if (iter != begin()) {
|
|
|
|
--iter;
|
|
|
|
const auto opcode = iter->opcode();
|
2017-10-30 21:42:26 +00:00
|
|
|
if (opcode == SpvOpLoopMerge || opcode == SpvOpSelectionMerge) {
|
2017-08-31 15:37:17 +00:00
|
|
|
result = &*iter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Instruction* BasicBlock::GetLoopMergeInst() const {
|
|
|
|
if (auto* merge = GetMergeInst()) {
|
|
|
|
if (merge->opcode() == SpvOpLoopMerge) {
|
|
|
|
return merge;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
Instruction* BasicBlock::GetLoopMergeInst() {
|
|
|
|
if (auto* merge = GetMergeInst()) {
|
|
|
|
if (merge->opcode() == SpvOpLoopMerge) {
|
|
|
|
return merge;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2016-11-10 17:11:50 +00:00
|
|
|
void BasicBlock::ForEachSuccessorLabel(
|
|
|
|
const std::function<void(const uint32_t)>& f) {
|
2017-10-13 18:25:21 +00:00
|
|
|
const auto br = &insts_.back();
|
2016-11-10 17:11:50 +00:00
|
|
|
switch (br->opcode()) {
|
|
|
|
case SpvOpBranch: {
|
|
|
|
f(br->GetOperand(0).words[0]);
|
|
|
|
} break;
|
|
|
|
case SpvOpBranchConditional:
|
|
|
|
case SpvOpSwitch: {
|
|
|
|
bool is_first = true;
|
|
|
|
br->ForEachInId([&is_first, &f](const uint32_t* idp) {
|
|
|
|
if (!is_first) f(*idp);
|
|
|
|
is_first = false;
|
|
|
|
});
|
|
|
|
} break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-23 23:05:38 +00:00
|
|
|
void BasicBlock::ForMergeAndContinueLabel(
|
|
|
|
const std::function<void(const uint32_t)>& f) {
|
|
|
|
auto ii = insts_.end();
|
|
|
|
--ii;
|
|
|
|
if (ii == insts_.begin()) return;
|
|
|
|
--ii;
|
2017-10-30 21:42:26 +00:00
|
|
|
if (ii->opcode() == SpvOpSelectionMerge || ii->opcode() == SpvOpLoopMerge) {
|
|
|
|
ii->ForEachInId([&f](const uint32_t* idp) { f(*idp); });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t BasicBlock::MergeBlockIdIfAny() const {
|
|
|
|
auto merge_ii = cend();
|
|
|
|
--merge_ii;
|
|
|
|
uint32_t mbid = 0;
|
|
|
|
if (merge_ii != cbegin()) {
|
|
|
|
--merge_ii;
|
|
|
|
if (merge_ii->opcode() == SpvOpLoopMerge) {
|
|
|
|
mbid = merge_ii->GetSingleWordInOperand(kLoopMergeMergeBlockIdInIdx);
|
|
|
|
} else if (merge_ii->opcode() == SpvOpSelectionMerge) {
|
|
|
|
mbid = merge_ii->GetSingleWordInOperand(kSelectionMergeMergeBlockIdInIdx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return mbid;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t BasicBlock::ContinueBlockIdIfAny() const {
|
|
|
|
auto merge_ii = cend();
|
|
|
|
--merge_ii;
|
|
|
|
uint32_t cbid = 0;
|
|
|
|
if (merge_ii != cbegin()) {
|
|
|
|
--merge_ii;
|
|
|
|
if (merge_ii->opcode() == SpvOpLoopMerge) {
|
|
|
|
cbid = merge_ii->GetSingleWordInOperand(kLoopMergeContinueBlockIdInIdx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return cbid;
|
2017-08-23 23:05:38 +00:00
|
|
|
}
|
|
|
|
|
2016-11-10 17:11:50 +00:00
|
|
|
} // namespace ir
|
|
|
|
} // namespace spvtools
|