2017-07-14 05:16:01 +08:00
|
|
|
//===- CFGBuilder.h - CFG building and updating utility ----------*- C++ -*-==//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2017-07-14 05:16:01 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
/// \file
|
|
|
|
/// CFGBuilders provides utilities fo building and updating CFG for testing
|
|
|
|
/// purposes.
|
|
|
|
///
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef LLVM_UNITTESTS_CFG_BUILDER_H
|
|
|
|
#define LLVM_UNITTESTS_CFG_BUILDER_H
|
|
|
|
|
|
|
|
#include "llvm/ADT/DenseMap.h"
|
|
|
|
#include "llvm/ADT/Optional.h"
|
|
|
|
#include "llvm/ADT/StringMap.h"
|
|
|
|
#include "llvm/ADT/StringRef.h"
|
|
|
|
#include "llvm/Support/Debug.h"
|
|
|
|
|
|
|
|
#include <memory>
|
|
|
|
#include <set>
|
2017-07-14 07:11:57 +08:00
|
|
|
#include <tuple>
|
2017-07-14 05:16:01 +08:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
|
|
|
class LLVMContext;
|
|
|
|
class Module;
|
|
|
|
class Function;
|
|
|
|
class BasicBlock;
|
|
|
|
class raw_ostream;
|
|
|
|
|
|
|
|
struct CFGHolder {
|
|
|
|
std::unique_ptr<LLVMContext> Context;
|
|
|
|
std::unique_ptr<Module> M;
|
|
|
|
Function *F;
|
|
|
|
|
|
|
|
CFGHolder(StringRef ModuleName = "m", StringRef FunctionName = "foo");
|
|
|
|
~CFGHolder(); // Defined in the .cpp file so we can use forward declarations.
|
|
|
|
};
|
|
|
|
|
|
|
|
/// \brief
|
|
|
|
/// CFGBuilder builds IR with specific CFG, based on the supplied list of arcs.
|
|
|
|
/// It's able to apply the provided updates and automatically modify the IR.
|
|
|
|
///
|
|
|
|
/// Internally it makes every basic block end with either SwitchInst or with
|
|
|
|
/// UnreachableInst. When all arc to a BB are deleted, the BB remains in the
|
|
|
|
/// function and doesn't get deleted.
|
|
|
|
///
|
|
|
|
class CFGBuilder {
|
|
|
|
public:
|
|
|
|
struct Arc {
|
|
|
|
StringRef From;
|
|
|
|
StringRef To;
|
|
|
|
|
2017-07-14 07:11:57 +08:00
|
|
|
friend bool operator<(const Arc &LHS, const Arc &RHS) {
|
|
|
|
return std::tie(LHS.From, LHS.To) <
|
|
|
|
std::tie(RHS.From, RHS.To);
|
|
|
|
}
|
2017-07-14 05:16:01 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
enum class ActionKind { Insert, Delete };
|
|
|
|
struct Update {
|
|
|
|
ActionKind Action;
|
2017-07-14 05:52:56 +08:00
|
|
|
Arc Edge;
|
2017-07-14 05:16:01 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
CFGBuilder(Function *F, const std::vector<Arc> &InitialArcs,
|
|
|
|
std::vector<Update> Updates);
|
|
|
|
|
|
|
|
BasicBlock *getOrAddBlock(StringRef BlockName);
|
|
|
|
Optional<Update> getNextUpdate() const;
|
|
|
|
Optional<Update> applyUpdate();
|
|
|
|
void dump(raw_ostream &OS = dbgs()) const;
|
|
|
|
|
|
|
|
private:
|
|
|
|
void buildCFG(const std::vector<Arc> &Arcs);
|
|
|
|
bool connect(const Arc &A);
|
|
|
|
bool disconnect(const Arc &A);
|
|
|
|
|
|
|
|
Function *F;
|
|
|
|
unsigned UpdateIdx = 0;
|
|
|
|
StringMap<BasicBlock *> NameToBlock;
|
|
|
|
std::set<Arc> Arcs;
|
|
|
|
std::vector<Update> Updates;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace llvm
|
|
|
|
|
|
|
|
#endif
|