blob: d1729db6e10d8d52804f7ff51a75c523c6caf962 [file] [log] [blame]
// Copyright 2017 The Clspv Authors. All rights reserved.
//
// 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 <deque>
#include "llvm/ADT/DenseSet.h"
#include "llvm/Analysis/LoopInfo.h"
#include "llvm/IR/Dominators.h"
#include "llvm/Pass.h"
#include "llvm/Support/raw_ostream.h"
#include "clspv/Option.h"
#include "ComputeStructuredOrder.h"
#include "Passes.h"
using namespace llvm;
#define DEBUG_TYPE "reorderbbs"
namespace {
struct ReorderBasicBlocksPass : public FunctionPass {
static char ID;
ReorderBasicBlocksPass() : FunctionPass(ID) {}
void getAnalysisUsage(AnalysisUsage &AU) const override {
AU.addRequired<DominatorTreeWrapperPass>();
AU.addRequired<LoopInfoWrapperPass>();
}
bool runOnFunction(Function &F) override;
};
} // namespace
char ReorderBasicBlocksPass::ID = 0;
INITIALIZE_PASS(ReorderBasicBlocksPass, "ReorderBasicBlocks",
"Reorder Basic Blocks Pass", false, false)
namespace clspv {
FunctionPass *createReorderBasicBlocksPass() {
return new ReorderBasicBlocksPass();
}
} // namespace clspv
bool ReorderBasicBlocksPass::runOnFunction(Function &F) {
bool Changed = false;
DominatorTree &DT = getAnalysis<DominatorTreeWrapperPass>().getDomTree();
if (clspv::Option::HackBlockOrder()) {
// Order basic blocks according to structured order. Structured subgraphs
// will be ordered contiguously within the binary.
//
// Assumes CFG has been structurized.
const LoopInfo &LI = getAnalysis<LoopInfoWrapperPass>().getLoopInfo();
std::deque<BasicBlock *> order;
DenseSet<BasicBlock *> visited;
clspv::ComputeStructuredOrder(&*F.begin(), &DT, LI, &order, &visited);
for (unsigned i = 1; i != order.size(); ++i) {
order[i]->moveAfter(order[i - 1]);
}
} else {
// spirv-val wants the order of basic blocks to follow dominance relation.
// Reorder basic blocks according to dominance relation.
// Traverse dominator tree using depth first order. Reorder basic blocks
// according to dominance relation.
for (auto II = df_begin(DT.getRootNode()), IE = df_end(DT.getRootNode());
II != IE; ++II) {
BasicBlock *BB = (*II)->getBlock();
BB->moveAfter(&F.back());
}
}
return Changed;
}