2018-01-29 10:39:55 +00:00
|
|
|
// Copyright (c) 2018 Google LLC.
|
|
|
|
//
|
|
|
|
// 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 "opt/licm_pass.h"
|
|
|
|
#include "opt/module.h"
|
|
|
|
#include "opt/pass.h"
|
|
|
|
|
|
|
|
#include <queue>
|
|
|
|
#include <utility>
|
|
|
|
|
|
|
|
namespace spvtools {
|
|
|
|
namespace opt {
|
|
|
|
|
2018-07-09 15:32:29 +00:00
|
|
|
Pass::Status LICMPass::Process(opt::IRContext* c) {
|
2018-01-29 10:39:55 +00:00
|
|
|
InitializeProcessing(c);
|
|
|
|
bool modified = false;
|
|
|
|
|
|
|
|
if (c != nullptr) {
|
|
|
|
modified = ProcessIRContext();
|
|
|
|
}
|
|
|
|
|
|
|
|
return modified ? Status::SuccessWithChange : Status::SuccessWithoutChange;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LICMPass::ProcessIRContext() {
|
|
|
|
bool modified = false;
|
2018-07-09 15:32:29 +00:00
|
|
|
opt::Module* module = get_module();
|
2018-01-29 10:39:55 +00:00
|
|
|
|
|
|
|
// Process each function in the module
|
2018-07-09 15:32:29 +00:00
|
|
|
for (opt::Function& f : *module) {
|
2018-01-29 10:39:55 +00:00
|
|
|
modified |= ProcessFunction(&f);
|
|
|
|
}
|
|
|
|
return modified;
|
|
|
|
}
|
|
|
|
|
2018-07-09 15:32:29 +00:00
|
|
|
bool LICMPass::ProcessFunction(opt::Function* f) {
|
2018-01-29 10:39:55 +00:00
|
|
|
bool modified = false;
|
2018-07-09 15:32:29 +00:00
|
|
|
opt::LoopDescriptor* loop_descriptor = context()->GetLoopDescriptor(f);
|
2018-01-29 10:39:55 +00:00
|
|
|
|
|
|
|
// Process each loop in the function
|
2018-07-09 15:32:29 +00:00
|
|
|
for (opt::Loop& loop : *loop_descriptor) {
|
2018-01-29 10:39:55 +00:00
|
|
|
// Ignore nested loops, as we will process them in order in ProcessLoop
|
|
|
|
if (loop.IsNested()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
modified |= ProcessLoop(&loop, f);
|
|
|
|
}
|
|
|
|
return modified;
|
|
|
|
}
|
|
|
|
|
2018-07-09 15:32:29 +00:00
|
|
|
bool LICMPass::ProcessLoop(opt::Loop* loop, opt::Function* f) {
|
2018-01-29 10:39:55 +00:00
|
|
|
bool modified = false;
|
|
|
|
|
|
|
|
// Process all nested loops first
|
2018-07-09 15:32:29 +00:00
|
|
|
for (opt::Loop* nested_loop : *loop) {
|
2018-01-29 10:39:55 +00:00
|
|
|
modified |= ProcessLoop(nested_loop, f);
|
|
|
|
}
|
|
|
|
|
2018-07-09 15:32:29 +00:00
|
|
|
std::vector<opt::BasicBlock*> loop_bbs{};
|
2018-01-29 10:39:55 +00:00
|
|
|
modified |= AnalyseAndHoistFromBB(loop, f, loop->GetHeaderBlock(), &loop_bbs);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < loop_bbs.size(); ++i) {
|
2018-07-09 15:32:29 +00:00
|
|
|
opt::BasicBlock* bb = loop_bbs[i];
|
2018-01-29 10:39:55 +00:00
|
|
|
// do not delete the element
|
|
|
|
modified |= AnalyseAndHoistFromBB(loop, f, bb, &loop_bbs);
|
|
|
|
}
|
|
|
|
|
|
|
|
return modified;
|
|
|
|
}
|
|
|
|
|
2018-07-09 15:32:29 +00:00
|
|
|
bool LICMPass::AnalyseAndHoistFromBB(opt::Loop* loop, opt::Function* f,
|
|
|
|
opt::BasicBlock* bb,
|
|
|
|
std::vector<opt::BasicBlock*>* loop_bbs) {
|
2018-01-29 10:39:55 +00:00
|
|
|
bool modified = false;
|
2018-07-09 15:32:29 +00:00
|
|
|
std::function<void(opt::Instruction*)> hoist_inst =
|
|
|
|
[this, &loop, &modified](opt::Instruction* inst) {
|
2018-01-29 10:39:55 +00:00
|
|
|
if (loop->ShouldHoistInstruction(this->context(), inst)) {
|
|
|
|
HoistInstruction(loop, inst);
|
|
|
|
modified = true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
if (IsImmediatelyContainedInLoop(loop, f, bb)) {
|
|
|
|
bb->ForEachInst(hoist_inst, false);
|
|
|
|
}
|
|
|
|
|
2018-04-20 16:28:40 +00:00
|
|
|
opt::DominatorAnalysis* dom_analysis = context()->GetDominatorAnalysis(f);
|
2018-01-29 10:39:55 +00:00
|
|
|
opt::DominatorTree& dom_tree = dom_analysis->GetDomTree();
|
|
|
|
|
|
|
|
for (opt::DominatorTreeNode* child_dom_tree_node :
|
|
|
|
*dom_tree.GetTreeNode(bb)) {
|
|
|
|
if (loop->IsInsideLoop(child_dom_tree_node->bb_)) {
|
|
|
|
loop_bbs->push_back(child_dom_tree_node->bb_);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return modified;
|
|
|
|
}
|
|
|
|
|
2018-07-09 15:32:29 +00:00
|
|
|
bool LICMPass::IsImmediatelyContainedInLoop(opt::Loop* loop, opt::Function* f,
|
|
|
|
opt::BasicBlock* bb) {
|
|
|
|
opt::LoopDescriptor* loop_descriptor = context()->GetLoopDescriptor(f);
|
2018-01-29 10:39:55 +00:00
|
|
|
return loop == (*loop_descriptor)[bb->id()];
|
|
|
|
}
|
|
|
|
|
2018-07-09 15:32:29 +00:00
|
|
|
void LICMPass::HoistInstruction(opt::Loop* loop, opt::Instruction* inst) {
|
|
|
|
opt::BasicBlock* pre_header_bb = loop->GetOrCreatePreHeaderBlock();
|
2018-01-29 10:39:55 +00:00
|
|
|
inst->InsertBefore(std::move(&(*pre_header_bb->tail())));
|
|
|
|
context()->set_instr_block(inst, pre_header_bb);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace opt
|
|
|
|
} // namespace spvtools
|